From 0cf9bf44a746681fd62262e1502f2320cae02984 Mon Sep 17 00:00:00 2001 From: wireswiss Date: Thu, 21 Jul 2016 20:33:11 +0300 Subject: [PATCH] Initial commit --- .bowerrc | 3 + .editorconfig | 9 + .gitattributes | 15 + .gitignore | 38 + .idea/codeStyleSettings.xml | 46 + .idea/modules.xml | 8 + .idea/wire-webapp.iml | 32 + .travis.yml | 42 + Gruntfile.coffee | 146 ++ LICENSE | 674 ++++++ README.md | 48 + app/audio/buzzer/applause.mp3 | Bin 0 -> 84038 bytes app/audio/buzzer/baby-cry.mp3 | Bin 0 -> 122912 bytes app/audio/buzzer/crowd-negative.mp3 | Bin 0 -> 60844 bytes app/audio/buzzer/dogs.mp3 | Bin 0 -> 59590 bytes app/audio/buzzer/gun-shot.mp3 | Bin 0 -> 84666 bytes app/audio/buzzer/wrong-answer.mp3 | Bin 0 -> 58666 bytes app/audio/digits/0.mp3 | Bin 0 -> 3240 bytes app/audio/digits/1.mp3 | Bin 0 -> 2592 bytes app/audio/digits/2.mp3 | Bin 0 -> 15488 bytes app/audio/digits/3.mp3 | Bin 0 -> 2592 bytes app/audio/digits/4.mp3 | Bin 0 -> 2808 bytes app/audio/digits/5.mp3 | Bin 0 -> 2808 bytes app/audio/digits/6.mp3 | Bin 0 -> 2808 bytes app/audio/digits/7.mp3 | Bin 0 -> 3240 bytes app/audio/digits/8.mp3 | Bin 0 -> 2376 bytes app/audio/digits/9.mp3 | Bin 0 -> 3024 bytes app/demo/demo.html | 189 ++ app/demo/template/avatars.htm | 289 +++ app/demo/template/buttons.htm | 100 + app/demo/template/checkbox.htm | 4 + app/demo/template/colors.htm | 11 + app/demo/template/dots.htm | 7 + app/demo/template/grid.htm | 5 + app/demo/template/icons.htm | 1 + app/demo/template/media-controls.htm | 38 + app/demo/template/microbar.htm | 9 + app/demo/template/modal.htm | 15 + app/demo/template/popovers.htm | 20 + app/font/Wire.ttf | Bin 0 -> 8996 bytes app/image/debug/ping-fireworks.png | Bin 0 -> 33010 bytes app/image/debug/wooden-background.jpg | Bin 0 -> 293648 bytes app/image/favicon.ico | Bin 0 -> 18094 bytes app/image/icon/wire-icon-128.png | Bin 0 -> 3986 bytes app/image/icon/wire-icon-256.png | Bin 0 -> 9222 bytes app/image/icon/wire-icon-64.png | Bin 0 -> 1832 bytes app/image/logo/notification.png | Bin 0 -> 21809 bytes app/image/logo/wire-logo-120.png | Bin 0 -> 6467 bytes app/page/auth.html | 360 ++++ app/page/index.html | 26 + app/page/template/_deploy/app.htm | 1 + app/page/template/_deploy/auth.htm | 1 + app/page/template/_deploy/component.htm | 1 + app/page/template/_deploy/debug.htm | 55 + app/page/template/_deploy/style.htm | 1 + app/page/template/_deploy/vendor.htm | 1 + app/page/template/_dist/app.htm | 277 +++ app/page/template/_dist/auth.htm | 4 + app/page/template/_dist/component.htm | 8 + app/page/template/_dist/debug.htm | 61 + app/page/template/_dist/style.htm | 1 + app/page/template/_dist/vendor.htm | 44 + app/page/template/_prod/app.htm | 1 + app/page/template/_prod/auth.htm | 1 + app/page/template/_prod/component.htm | 1 + app/page/template/_prod/debug.htm | 0 app/page/template/_prod/style.htm | 1 + app/page/template/_prod/vendor.htm | 1 + .../conversation/connect-requests.htm | 18 + .../conversation/conversation-input.htm | 62 + .../conversation/conversation-titlebar.htm | 28 + .../conversation/conversation-verified.htm | 8 + .../template/conversation/conversation.htm | 22 + .../template/conversation/detail-view.htm | 6 + app/page/template/conversation/giphy.htm | 46 + .../template/conversation/message-list.htm | 18 + .../template/conversation/participants.htm | 74 + app/page/template/graph.htm | 5 + app/page/template/list/actions.htm | 60 + app/page/template/list/archive.htm | 57 + app/page/template/list/conversation-list.htm | 199 ++ app/page/template/list/start-ui.htm | 167 ++ app/page/template/loading.htm | 6 + app/page/template/meta.htm | 9 + app/page/template/modals.htm | 287 +++ .../template/partials/template-confirm.htm | 96 + .../template/partials/template-message.htm | 235 +++ .../partials/template-user-profile.htm | 201 ++ app/page/template/self/self-about.htm | 21 + app/page/template/self/self-profile.htm | 28 + app/page/template/self/settings-menu.htm | 8 + app/page/template/settings.htm | 173 ++ app/page/template/svg.htm | 84 + app/page/template/video-calling.htm | 99 + app/page/template/warning.htm | 153 ++ app/page/template/wire-main.htm | 59 + app/script/announce/AnnounceRepository.coffee | 75 + app/script/announce/AnnounceService.coffee | 35 + app/script/assets/Asset.coffee | 64 + app/script/assets/AssetCrypto.coffee | 84 + app/script/assets/AssetRemoteData.coffee | 80 + app/script/assets/AssetRetentionPolicy.coffee | 25 + app/script/assets/AssetService.coffee | 418 ++++ app/script/assets/AssetTransferState.coffee | 28 + app/script/assets/AssetType.coffee | 28 + .../assets/AssetUploadFailedReason.coffee | 25 + app/script/assets/ImageSizeType.coffee | 26 + app/script/audio/AudioPlayingType.coffee | 40 + app/script/audio/AudioRepository.coffee | 160 ++ app/script/audio/AudioType.coffee | 33 + app/script/auth/AccessTokenError.coffee | 36 + app/script/auth/AuthRepository.coffee | 255 +++ app/script/auth/AuthService.coffee | 303 +++ app/script/auth/AuthView.coffee | 62 + app/script/auth/URLParameter.coffee | 28 + app/script/auth/ValidationError.coffee | 28 + app/script/cache/CacheRepository.coffee | 56 + app/script/calling/CallCenter.coffee | 267 +++ app/script/calling/CallError.coffee | 42 + app/script/calling/CallService.coffee | 148 ++ app/script/calling/CallTrackingInfo.coffee | 34 + app/script/calling/entities/Call.coffee | 518 +++++ app/script/calling/entities/Flow.coffee | 946 +++++++++ app/script/calling/entities/FlowAudio.coffee | 120 ++ .../calling/entities/Participant.coffee | 58 + .../calling/enum/CallFinishedReason.coffee | 29 + app/script/calling/enum/CallState.coffee | 31 + .../calling/enum/CallStateEventCause.coffee | 27 + .../calling/enum/CallStateGroups.coffee | 43 + .../calling/enum/MediaDeviceType.coffee | 27 + .../calling/enum/MediaStreamSource.coffee | 25 + app/script/calling/enum/MediaType.coffee | 28 + .../calling/enum/ParticipantState.coffee | 25 + .../calling/enum/SDPNegotiationMode.coffee | 26 + app/script/calling/enum/SDPSource.coffee | 25 + .../calling/enum/VideoOrientation.coffee | 25 + .../handler/CallSignalingHandler.coffee | 333 +++ .../calling/handler/CallStateHandler.coffee | 748 +++++++ .../handler/MediaDevicesHandler.coffee | 255 +++ .../handler/MediaElementHandler.coffee | 109 + .../calling/handler/MediaStreamHandler.coffee | 625 ++++++ .../calling/mapper/ICECandidateMapper.coffee | 45 + app/script/calling/mapper/SDPMapper.coffee | 41 + .../calling/payloads/FlowDeletionInfo.coffee | 29 + .../calling/payloads/ICECandidateInfo.coffee | 34 + .../calling/payloads/MediaStreamInfo.coffee | 33 + app/script/calling/payloads/SDPInfo.coffee | 35 + .../calling/rtc/ICEConnectionState.coffee | 32 + .../calling/rtc/ICEGatheringState.coffee | 28 + .../calling/rtc/MediaStreamError.coffee | 37 + .../calling/rtc/MediaStreamErrorTypes.coffee | 42 + app/script/calling/rtc/SDPType.coffee | 31 + app/script/calling/rtc/SignalingState.coffee | 32 + app/script/calling/rtc/StatsType.coffee | 29 + app/script/client/Client.coffee | 75 + app/script/client/ClientError.coffee | 41 + app/script/client/ClientMapper.coffee | 60 + app/script/client/ClientRepository.coffee | 581 ++++++ app/script/client/ClientService.coffee | 193 ++ app/script/client/ClientType.coffee | 24 + .../components/accentColorPicker.coffee | 58 + app/script/components/asset/audioAsset.coffee | 116 ++ .../asset/controls/audioSeekBar.coffee | 100 + .../asset/controls/mediaButton.coffee | 85 + .../components/asset/controls/seekBar.coffee | 86 + app/script/components/asset/fileAsset.coffee | 112 + .../components/asset/linkPreviewAsset.coffee | 60 + .../components/asset/locationAsset.coffee | 39 + app/script/components/asset/videoAsset.coffee | 131 ++ .../components/calling/chooseScreen.coffee | 45 + .../calling/deviceToggleButton.coffee | 38 + app/script/components/commonContacts.coffee | 64 + app/script/components/deviceCard.coffee | 85 + app/script/components/deviceRemove.coffee | 69 + app/script/components/groupList.coffee | 52 + app/script/components/inputElement.coffee | 56 + app/script/components/topPeople.coffee | 49 + app/script/components/userAvatar.coffee | 89 + app/script/components/userInput.coffee | 67 + app/script/components/userList.coffee | 142 ++ app/script/components/userProfile.coffee | 269 +++ app/script/config.coffee | 112 + app/script/connect/ConnectError.coffee | 35 + .../connect/ConnectGoogleService.coffee | 129 ++ app/script/connect/ConnectRepository.coffee | 196 ++ app/script/connect/ConnectService.coffee | 42 + app/script/connect/ConnectSource.coffee | 24 + app/script/connect/ConnectTrigger.coffee | 25 + app/script/connect/PhoneBook.coffee | 30 + .../conversation/ConversationMapper.coffee | 156 ++ .../ConversationRepository.coffee | 1813 +++++++++++++++++ .../conversation/ConversationService.coffee | 420 ++++ .../conversation/ConversationStatus.coffee | 25 + .../conversation/ConversationType.coffee | 27 + .../ConversationUnreadType.coffee | 27 + .../ConversationUpdateType.coffee | 29 + app/script/conversation/EventMapper.coffee | 452 ++++ .../cryptography/CryptographyError.coffee | 38 + .../cryptography/CryptographyErrorType.coffee | 30 + .../cryptography/CryptographyMapper.coffee | 228 +++ .../CryptographyRepository.coffee | 625 ++++++ .../cryptography/CryptographyService.coffee | 55 + app/script/entity/Connection.coffee | 29 + app/script/entity/Conversation.coffee | 511 +++++ app/script/entity/User.coffee | 140 ++ app/script/entity/message/Asset.coffee | 121 ++ app/script/entity/message/CallMessage.coffee | 50 + .../entity/message/ContentMessage.coffee | 80 + .../entity/message/DecryptErrorMessage.coffee | 69 + app/script/entity/message/File.coffee | 128 ++ app/script/entity/message/LinkPreview.coffee | 32 + app/script/entity/message/Location.coffee | 33 + app/script/entity/message/MediumImage.coffee | 39 + .../entity/message/MemberMessage.coffee | 111 + app/script/entity/message/Message.coffee | 209 ++ .../entity/message/NewDeviceMessage.coffee | 61 + app/script/entity/message/PingMessage.coffee | 35 + app/script/entity/message/PreviewImage.coffee | 54 + .../entity/message/RenameMessage.coffee | 30 + .../entity/message/SystemMessage.coffee | 27 + app/script/entity/message/Text.coffee | 52 + app/script/event/Backend.coffee | 63 + app/script/event/Client.coffee | 24 + app/script/event/EventError.coffee | 37 + app/script/event/EventRepository.coffee | 393 ++++ app/script/event/EventTypeHandling.coffee | 43 + app/script/event/NotificationService.coffee | 88 + app/script/event/WebApp.coffee | 187 ++ app/script/event/WebSocketService.coffee | 186 ++ app/script/extension/GiphyContentSizes.coffee | 38 + app/script/extension/GiphyRepository.coffee | 143 ++ app/script/extension/GiphyService.coffee | 90 + app/script/links/LinkPreviewError.coffee | 33 + app/script/links/LinkPreviewHelpers.coffee | 45 + .../links/LinkPreviewProtoBuilder.coffee | 47 + app/script/links/LinkPreviewRepository.coffee | 89 + app/script/localization/Localizer.coffee | 106 + app/script/localization/strings-de.coffee | 566 +++++ app/script/localization/strings-init.coffee | 26 + app/script/localization/strings.coffee | 566 +++++ app/script/location/GeoLocation.coffee | 83 + app/script/main/app.coffee | 483 +++++ app/script/main/auth.coffee | 36 + app/script/main/auth_init_dist.coffee | 43 + app/script/main/auth_init_edge.coffee | 43 + app/script/main/auth_init_prod.coffee | 29 + app/script/main/auth_init_staging.coffee | 43 + app/script/media/MediaEmbeds.coffee | 236 +++ app/script/media/MediaParser.coffee | 52 + app/script/message/CallMessageType.coffee | 25 + app/script/message/PingMessageType.coffee | 25 + app/script/message/SuperType.coffee | 36 + app/script/message/SystemMessageType.coffee | 35 + app/script/search/SearchLevel.coffee | 28 + app/script/search/SearchMode.coffee | 28 + app/script/search/SearchRepository.coffee | 221 ++ app/script/search/SearchResultMapper.coffee | 46 + app/script/search/SearchService.coffee | 83 + app/script/service/BackendClientError.coffee | 78 + app/script/service/BackendEnvironment.coffee | 25 + app/script/service/Client.coffee | 242 +++ app/script/storage/SkipError.coffee | 29 + app/script/storage/StorageKey.coffee | 39 + app/script/storage/StorageRepository.coffee | 427 ++++ app/script/storage/StorageService.coffee | 339 +++ .../SystemNotificationRepository.coffee | 476 +++++ .../app_init/AppInitStatistics.coffee | 49 + .../app_init/AppInitStatisticsValue.coffee | 30 + .../app_init/AppInitTelemetry.coffee | 56 + .../telemetry/app_init/AppInitTimings.coffee | 46 + .../app_init/AppInitTimingsStep.coffee | 35 + .../telemetry/calling/AudioStreamStats.coffee | 28 + .../telemetry/calling/CallSetupSteps.coffee | 39 + .../calling/CallSetupStepsOrder.coffee | 57 + .../telemetry/calling/CallSetupTimings.coffee | 68 + .../telemetry/calling/CallTelemetry.coffee | 198 ++ .../telemetry/calling/ConnectionStats.coffee | 32 + .../telemetry/calling/FlowTelemetry.coffee | 458 +++++ .../telemetry/calling/MediaStreamStats.coffee | 32 + app/script/telemetry/calling/Stats.coffee | 29 + .../telemetry/calling/StreamStats.coffee | 27 + .../telemetry/calling/VideoStreamStats.coffee | 32 + app/script/tracking/EventName.coffee | 95 + .../tracking/EventTrackingRepository.coffee | 266 +++ app/script/tracking/SessionEventName.coffee | 49 + .../tracking/event/PhoneVerification.coffee | 36 + .../tracking/event/PictureTakenEvent.coffee | 35 + app/script/ui/Shortcut.coffee | 218 ++ app/script/ui/WindowHandler.coffee | 92 + app/script/user/ConnectionLevel.coffee | 28 + app/script/user/ConnectionStatus.coffee | 30 + app/script/user/UserConnectionMapper.coffee | 64 + app/script/user/UserError.coffee | 35 + app/script/user/UserMapper.coffee | 112 + app/script/user/UserProperties.coffee | 39 + app/script/user/UserRepository.coffee | 694 +++++++ app/script/user/UserService.coffee | 315 +++ app/script/util/ArrayUtil.coffee | 72 + app/script/util/BrowserPermissionType.coffee | 27 + app/script/util/Comparison.coffee | 31 + app/script/util/CountryCodes.coffee | 1511 ++++++++++++++ app/script/util/Crypto.coffee | 42 + app/script/util/DebugUtil.coffee | 50 + app/script/util/Environment.coffee | 133 ++ app/script/util/Invite.coffee | 162 ++ app/script/util/Statistics.coffee | 60 + app/script/util/Time.coffee | 42 + app/script/util/TimeTracker.coffee | 49 + app/script/util/confirm.coffee | 80 + app/script/util/keycode.coffee | 31 + app/script/util/protobuf.coffee | 32 + app/script/util/scroll-helpers.coffee | 58 + app/script/util/shims.coffee | 29 + app/script/util/storage.coffee | 39 + app/script/util/types.coffee | 31 + app/script/util/util.coffee | 685 +++++++ app/script/view_model/ActionsViewModel.coffee | 125 ++ app/script/view_model/ArchiveViewModel.coffee | 72 + app/script/view_model/AuthViewModel.coffee | 1311 ++++++++++++ .../view_model/BackgroundViewModel.coffee | 63 + .../view_model/CallShortcutsViewModel.coffee | 70 + .../ConnectRequestsViewModel.coffee | 52 + .../ConversationInputViewModel.coffee | 170 ++ .../ConversationListViewModel.coffee | 202 ++ .../ConversationTitlebarViewModel.coffee | 69 + app/script/view_model/DebugViewModel.coffee | 63 + app/script/view_model/GiphyViewModel.coffee | 143 ++ .../ImageDetailViewViewModel.coffee | 73 + app/script/view_model/LoadingViewModel.coffee | 63 + app/script/view_model/MainViewModel.coffee | 47 + .../view_model/MessageListViewModel.coffee | 471 +++++ app/script/view_model/ModalsViewModel.coffee | 210 ++ .../view_model/ParticipantsViewModel.coffee | 238 +++ app/script/view_model/RightViewModel.coffee | 176 ++ .../view_model/SelfProfileViewModel.coffee | 165 ++ .../view_model/SettingsViewModel.coffee | 194 ++ app/script/view_model/StartUIViewModel.coffee | 486 +++++ .../view_model/VideoCallingViewModel.coffee | 212 ++ .../view_model/WarningsViewModel.coffee | 110 + app/script/view_model/WelcomeViewModel.coffee | 63 + .../view_model/WindowTitleViewModel.coffee | 68 + .../view_model/bindings/CommonBindings.coffee | 287 +++ .../bindings/ConversationListBindings.coffee | 65 + .../bindings/ListBackgroundBindings.coffee | 67 + .../bindings/MessageListBindings.coffee | 172 ++ .../view_model/bindings/SearchBindings.coffee | 41 + app/style/auth/account.less | 132 ++ app/style/auth/animations.less | 58 + app/style/auth/auth.less | 59 + app/style/auth/base.less | 169 ++ app/style/auth/limit.less | 40 + app/style/auth/posted.less | 64 + app/style/auth/verify.less | 54 + app/style/base.less | 53 + app/style/common/animations.less | 45 + app/style/common/bubble.less | 51 + app/style/common/button-round.less | 226 ++ app/style/common/button.less | 213 ++ app/style/common/checkbox.less | 70 + app/style/common/colors.less | 52 + app/style/common/common.less | 38 + app/style/common/hide-controls.less | 28 + app/style/common/hint.less | 89 + app/style/common/input.less | 115 ++ app/style/common/label.less | 47 + app/style/common/link.less | 29 + app/style/common/mixins.less | 375 ++++ app/style/common/modal.less | 157 ++ app/style/common/shapes.less | 46 + app/style/common/slider.less | 62 + app/style/common/spinner.less | 53 + app/style/common/three-dots.less | 61 + app/style/common/variables.less | 274 +++ app/style/components/accent-color-picker.less | 66 + app/style/components/asset/audio-asset.less | 37 + app/style/components/asset/common/common.less | 30 + .../asset/controls/audio-seek-bar.less | 36 + .../asset/controls/media-button.less | 93 + .../components/asset/controls/seek-bar.less | 56 + app/style/components/asset/file-asset.less | 68 + .../components/asset/link-preview-asset.less | 68 + .../components/asset/location-asset.less | 36 + app/style/components/asset/video-asset.less | 110 + .../components/calling/choose-screen.less | 79 + .../calling/device-toggle-button.less | 44 + app/style/components/common-contacts.less | 69 + app/style/components/device-card.less | 68 + app/style/components/device-remove.less | 73 + app/style/components/group-list.less | 28 + app/style/components/input-element.less | 64 + app/style/components/search-list.less | 196 ++ app/style/components/user-avatar.less | 258 +++ app/style/components/user-input.less | 117 ++ app/style/components/user-list.less | 30 + app/style/components/user-profile.less | 286 +++ app/style/conversation/confirm.less | 50 + app/style/conversation/connect-requests.less | 89 + .../conversation/conversation-input.less | 85 + .../conversation/conversation-titlebar.less | 60 + app/style/conversation/conversation.less | 55 + app/style/conversation/detail-view.less | 56 + app/style/conversation/giphy.less | 81 + app/style/conversation/message-list.less | 400 ++++ app/style/conversation/participants.less | 207 ++ app/style/debug.less | 165 ++ app/style/fonts/README.md | 6 + app/style/fonts/zeta-neue.css | 215 ++ app/style/framework.less | 56 + app/style/list/actions.less | 23 + app/style/list/archive.less | 51 + app/style/list/background.less | 89 + app/style/list/conversation-list-call.less | 82 + app/style/list/conversation-list.less | 161 ++ app/style/list/invite-bubble.less | 115 ++ app/style/list/start-ui.less | 204 ++ app/style/loading-screen.less | 62 + app/style/main.less | 98 + app/style/self/self-about.less | 58 + app/style/self/self-profile.less | 91 + app/style/self/self-settings.less | 178 ++ app/style/support.less | 96 + app/style/video-calling.less | 165 ++ app/style/warnings.less | 102 + app/style/welcome.less | 56 + aws/application.py | 165 ++ aws/config.py | 66 + aws/libs/__init__.py | 0 aws/libs/flask_sslify.py | 74 + aws/libs/httpagentparser/__init__.py | 675 ++++++ aws/libs/httpagentparser/more.py | 29 + aws/main.py | 155 ++ aws/requirements.txt | 2 + aws/templates/robots.txt | 4 + aws/templates/sitemap.xml | 14 + aws/util.py | 186 ++ bin/country_codes.py | 52 + bower.json | 168 ++ coffeelint.json | 161 ++ crowdin.yaml | 9 + grunt/config/aws_s3.coffee | 34 + grunt/config/bower.coffee | 31 + grunt/config/clean.coffee | 54 + grunt/config/codo.coffee | 28 + grunt/config/coffee.coffee | 88 + grunt/config/coffeelint.coffee | 28 + grunt/config/compress.coffee | 32 + grunt/config/connect.coffee | 26 + grunt/config/copy.coffee | 93 + grunt/config/includereplace.coffee | 135 ++ grunt/config/karma.coffee | 37 + grunt/config/less.coffee | 45 + grunt/config/open.coffee | 27 + grunt/config/postcss.coffee | 40 + grunt/config/shell.coffee | 28 + grunt/config/todo.coffee | 63 + grunt/config/uglify.coffee | 35 + grunt/config/watch.coffee | 41 + grunt/tasks/aws.coffee | 49 + grunt/tasks/misc.coffee | 165 ++ grunt/tasks/raygun.coffee | 77 + grunt/tasks/scripts.coffee | 46 + karma.conf.js | 119 ++ package.json | 51 + test/api/OpenGraphMocks.js | 82 + test/api/SDP_payloads.js | 215 ++ test/api/TestFactory.js | 307 +++ test/api/calling-payloads/call.flow-active.md | 19 + test/api/calling-payloads/call.flow-add.md | 24 + test/api/calling-payloads/call.flow-delete.md | 9 + test/api/calling-payloads/call.remote-sdp.md | 11 + test/api/calling-payloads/call.state.md | 77 + .../conversation.voice-channel-activate.md | 12 + .../conversation.voice-channel-deactivate.md | 14 + test/api/environment.js | 33 + test/api/payloads.js | 287 +++ .../announce/AnnounceRepositorySpec.coffee | 82 + .../announce/AnnounceServiceSpec.coffee | 98 + test/unit_tests/assets/AssetCryptoSpec.coffee | 32 + .../assets/AssetRemoteDataSpec.coffee | 41 + .../unit_tests/auth/AuthRepositorySpec.coffee | 70 + .../cache/CacheRepositorySpec.coffee | 43 + test/unit_tests/calling/CallCenterSpec.coffee | 444 ++++ .../calling/CallRequestResponseMock.coffee | 102 + .../calling/entities/FlowSpec.coffee | 62 + .../calling/mapper/SDPMapperSpec.coffee | 37 + .../unit_tests/client/ClientMapperSpec.coffee | 158 ++ .../client/ClientRepositorySpec.coffee | 220 ++ test/unit_tests/client/ClientSpec.coffee | 32 + .../connect/ConnectGoogleServiceSpec.coffee | 82 + .../ConversationMapperSpec.coffee | 168 ++ .../ConversationRepositorySpec.coffee | 645 ++++++ .../ConversationServiceSpec.coffee | 58 + .../conversation/EventMapperSpec.coffee | 93 + .../CryptographyMapperSpec.coffee | 418 ++++ .../CryptographyRepositorySpec.coffee | 118 ++ .../unit_tests/entity/ConversationSpec.coffee | 511 +++++ test/unit_tests/entity/UserSpec.coffee | 199 ++ .../unit_tests/entity/message/FileSpec.coffee | 47 + .../entity/message/MediumImageSpec.coffee | 46 + .../entity/message/MemberMessageSpec.coffee | 109 + .../entity/message/MessageEntitiesSpec.coffee | 76 + .../event/EventRepositorySpec.coffee | 151 ++ .../extension/GiphyRepositorySpec.coffee | 72 + .../links/LinkPreviewHelpersSpec.coffee | 67 + .../links/LinkPreviewProtoBuilderSpec.coffee | 84 + .../links/LinkPreviewRepositorySpec.coffee | 50 + .../localization/LocalizerSpec.coffee | 48 + .../location/GeoLocationSpec.coffee | 35 + test/unit_tests/media/MediaEmbedsSpec.coffee | 389 ++++ .../search/SearchRepositorySpec.coffee | 88 + test/unit_tests/service/ClientSpec.coffee | 198 ++ .../SystemNotificationRepositorySpec.coffee | 406 ++++ .../EventTrackingRepositorySpec.coffee | 153 ++ test/unit_tests/ui/ShortcutSpec.coffee | 97 + test/unit_tests/user/UserMapperSpec.coffee | 102 + .../unit_tests/user/UserRepositorySpec.coffee | 240 +++ test/unit_tests/user/UserServiceSpec.coffee | 149 ++ test/unit_tests/util/ArrayUtilSpec.coffee | 33 + test/unit_tests/util/ComparisonSpec.coffee | 70 + test/unit_tests/util/CountryCodesSpec.coffee | 51 + test/unit_tests/util/CryptoSpec.coffee | 67 + test/unit_tests/util/DebugUtilSpec.coffee | 51 + test/unit_tests/util/EnvironmentSpec.coffee | 25 + test/unit_tests/util/InviteSpec.coffee | 56 + test/unit_tests/util/TimeSpec.coffee | 57 + test/unit_tests/util/UtilSpec.coffee | 1044 ++++++++++ test/unit_tests/util/bindingsSpec.coffee | 161 ++ trans.py | 49 + 528 files changed, 58760 insertions(+) create mode 100644 .bowerrc create mode 100644 .editorconfig create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .idea/codeStyleSettings.xml create mode 100644 .idea/modules.xml create mode 100644 .idea/wire-webapp.iml create mode 100644 .travis.yml create mode 100755 Gruntfile.coffee create mode 100644 LICENSE create mode 100644 README.md create mode 100644 app/audio/buzzer/applause.mp3 create mode 100644 app/audio/buzzer/baby-cry.mp3 create mode 100644 app/audio/buzzer/crowd-negative.mp3 create mode 100644 app/audio/buzzer/dogs.mp3 create mode 100644 app/audio/buzzer/gun-shot.mp3 create mode 100644 app/audio/buzzer/wrong-answer.mp3 create mode 100644 app/audio/digits/0.mp3 create mode 100644 app/audio/digits/1.mp3 create mode 100644 app/audio/digits/2.mp3 create mode 100644 app/audio/digits/3.mp3 create mode 100644 app/audio/digits/4.mp3 create mode 100644 app/audio/digits/5.mp3 create mode 100644 app/audio/digits/6.mp3 create mode 100644 app/audio/digits/7.mp3 create mode 100644 app/audio/digits/8.mp3 create mode 100644 app/audio/digits/9.mp3 create mode 100644 app/demo/demo.html create mode 100644 app/demo/template/avatars.htm create mode 100644 app/demo/template/buttons.htm create mode 100644 app/demo/template/checkbox.htm create mode 100644 app/demo/template/colors.htm create mode 100644 app/demo/template/dots.htm create mode 100644 app/demo/template/grid.htm create mode 100644 app/demo/template/icons.htm create mode 100644 app/demo/template/media-controls.htm create mode 100644 app/demo/template/microbar.htm create mode 100644 app/demo/template/modal.htm create mode 100644 app/demo/template/popovers.htm create mode 100755 app/font/Wire.ttf create mode 100644 app/image/debug/ping-fireworks.png create mode 100644 app/image/debug/wooden-background.jpg create mode 100644 app/image/favicon.ico create mode 100755 app/image/icon/wire-icon-128.png create mode 100755 app/image/icon/wire-icon-256.png create mode 100755 app/image/icon/wire-icon-64.png create mode 100644 app/image/logo/notification.png create mode 100644 app/image/logo/wire-logo-120.png create mode 100644 app/page/auth.html create mode 100644 app/page/index.html create mode 100644 app/page/template/_deploy/app.htm create mode 100644 app/page/template/_deploy/auth.htm create mode 100644 app/page/template/_deploy/component.htm create mode 100644 app/page/template/_deploy/debug.htm create mode 100644 app/page/template/_deploy/style.htm create mode 100644 app/page/template/_deploy/vendor.htm create mode 100644 app/page/template/_dist/app.htm create mode 100644 app/page/template/_dist/auth.htm create mode 100644 app/page/template/_dist/component.htm create mode 100644 app/page/template/_dist/debug.htm create mode 100644 app/page/template/_dist/style.htm create mode 100644 app/page/template/_dist/vendor.htm create mode 100644 app/page/template/_prod/app.htm create mode 100644 app/page/template/_prod/auth.htm create mode 100644 app/page/template/_prod/component.htm create mode 100644 app/page/template/_prod/debug.htm create mode 100644 app/page/template/_prod/style.htm create mode 100644 app/page/template/_prod/vendor.htm create mode 100644 app/page/template/conversation/connect-requests.htm create mode 100644 app/page/template/conversation/conversation-input.htm create mode 100644 app/page/template/conversation/conversation-titlebar.htm create mode 100644 app/page/template/conversation/conversation-verified.htm create mode 100644 app/page/template/conversation/conversation.htm create mode 100644 app/page/template/conversation/detail-view.htm create mode 100644 app/page/template/conversation/giphy.htm create mode 100644 app/page/template/conversation/message-list.htm create mode 100644 app/page/template/conversation/participants.htm create mode 100644 app/page/template/graph.htm create mode 100644 app/page/template/list/actions.htm create mode 100644 app/page/template/list/archive.htm create mode 100644 app/page/template/list/conversation-list.htm create mode 100644 app/page/template/list/start-ui.htm create mode 100644 app/page/template/loading.htm create mode 100644 app/page/template/meta.htm create mode 100644 app/page/template/modals.htm create mode 100644 app/page/template/partials/template-confirm.htm create mode 100644 app/page/template/partials/template-message.htm create mode 100644 app/page/template/partials/template-user-profile.htm create mode 100644 app/page/template/self/self-about.htm create mode 100644 app/page/template/self/self-profile.htm create mode 100644 app/page/template/self/settings-menu.htm create mode 100644 app/page/template/settings.htm create mode 100644 app/page/template/svg.htm create mode 100644 app/page/template/video-calling.htm create mode 100644 app/page/template/warning.htm create mode 100644 app/page/template/wire-main.htm create mode 100644 app/script/announce/AnnounceRepository.coffee create mode 100644 app/script/announce/AnnounceService.coffee create mode 100644 app/script/assets/Asset.coffee create mode 100644 app/script/assets/AssetCrypto.coffee create mode 100644 app/script/assets/AssetRemoteData.coffee create mode 100644 app/script/assets/AssetRetentionPolicy.coffee create mode 100644 app/script/assets/AssetService.coffee create mode 100644 app/script/assets/AssetTransferState.coffee create mode 100644 app/script/assets/AssetType.coffee create mode 100644 app/script/assets/AssetUploadFailedReason.coffee create mode 100644 app/script/assets/ImageSizeType.coffee create mode 100644 app/script/audio/AudioPlayingType.coffee create mode 100644 app/script/audio/AudioRepository.coffee create mode 100644 app/script/audio/AudioType.coffee create mode 100644 app/script/auth/AccessTokenError.coffee create mode 100644 app/script/auth/AuthRepository.coffee create mode 100644 app/script/auth/AuthService.coffee create mode 100644 app/script/auth/AuthView.coffee create mode 100644 app/script/auth/URLParameter.coffee create mode 100644 app/script/auth/ValidationError.coffee create mode 100644 app/script/cache/CacheRepository.coffee create mode 100644 app/script/calling/CallCenter.coffee create mode 100644 app/script/calling/CallError.coffee create mode 100644 app/script/calling/CallService.coffee create mode 100644 app/script/calling/CallTrackingInfo.coffee create mode 100644 app/script/calling/entities/Call.coffee create mode 100644 app/script/calling/entities/Flow.coffee create mode 100644 app/script/calling/entities/FlowAudio.coffee create mode 100644 app/script/calling/entities/Participant.coffee create mode 100644 app/script/calling/enum/CallFinishedReason.coffee create mode 100644 app/script/calling/enum/CallState.coffee create mode 100644 app/script/calling/enum/CallStateEventCause.coffee create mode 100644 app/script/calling/enum/CallStateGroups.coffee create mode 100644 app/script/calling/enum/MediaDeviceType.coffee create mode 100644 app/script/calling/enum/MediaStreamSource.coffee create mode 100644 app/script/calling/enum/MediaType.coffee create mode 100644 app/script/calling/enum/ParticipantState.coffee create mode 100644 app/script/calling/enum/SDPNegotiationMode.coffee create mode 100644 app/script/calling/enum/SDPSource.coffee create mode 100644 app/script/calling/enum/VideoOrientation.coffee create mode 100644 app/script/calling/handler/CallSignalingHandler.coffee create mode 100644 app/script/calling/handler/CallStateHandler.coffee create mode 100644 app/script/calling/handler/MediaDevicesHandler.coffee create mode 100644 app/script/calling/handler/MediaElementHandler.coffee create mode 100644 app/script/calling/handler/MediaStreamHandler.coffee create mode 100644 app/script/calling/mapper/ICECandidateMapper.coffee create mode 100644 app/script/calling/mapper/SDPMapper.coffee create mode 100644 app/script/calling/payloads/FlowDeletionInfo.coffee create mode 100644 app/script/calling/payloads/ICECandidateInfo.coffee create mode 100644 app/script/calling/payloads/MediaStreamInfo.coffee create mode 100644 app/script/calling/payloads/SDPInfo.coffee create mode 100644 app/script/calling/rtc/ICEConnectionState.coffee create mode 100644 app/script/calling/rtc/ICEGatheringState.coffee create mode 100644 app/script/calling/rtc/MediaStreamError.coffee create mode 100644 app/script/calling/rtc/MediaStreamErrorTypes.coffee create mode 100644 app/script/calling/rtc/SDPType.coffee create mode 100644 app/script/calling/rtc/SignalingState.coffee create mode 100644 app/script/calling/rtc/StatsType.coffee create mode 100644 app/script/client/Client.coffee create mode 100644 app/script/client/ClientError.coffee create mode 100644 app/script/client/ClientMapper.coffee create mode 100644 app/script/client/ClientRepository.coffee create mode 100644 app/script/client/ClientService.coffee create mode 100644 app/script/client/ClientType.coffee create mode 100644 app/script/components/accentColorPicker.coffee create mode 100644 app/script/components/asset/audioAsset.coffee create mode 100644 app/script/components/asset/controls/audioSeekBar.coffee create mode 100644 app/script/components/asset/controls/mediaButton.coffee create mode 100644 app/script/components/asset/controls/seekBar.coffee create mode 100644 app/script/components/asset/fileAsset.coffee create mode 100644 app/script/components/asset/linkPreviewAsset.coffee create mode 100644 app/script/components/asset/locationAsset.coffee create mode 100644 app/script/components/asset/videoAsset.coffee create mode 100644 app/script/components/calling/chooseScreen.coffee create mode 100644 app/script/components/calling/deviceToggleButton.coffee create mode 100644 app/script/components/commonContacts.coffee create mode 100644 app/script/components/deviceCard.coffee create mode 100644 app/script/components/deviceRemove.coffee create mode 100644 app/script/components/groupList.coffee create mode 100644 app/script/components/inputElement.coffee create mode 100644 app/script/components/topPeople.coffee create mode 100644 app/script/components/userAvatar.coffee create mode 100644 app/script/components/userInput.coffee create mode 100644 app/script/components/userList.coffee create mode 100644 app/script/components/userProfile.coffee create mode 100644 app/script/config.coffee create mode 100644 app/script/connect/ConnectError.coffee create mode 100644 app/script/connect/ConnectGoogleService.coffee create mode 100644 app/script/connect/ConnectRepository.coffee create mode 100644 app/script/connect/ConnectService.coffee create mode 100644 app/script/connect/ConnectSource.coffee create mode 100644 app/script/connect/ConnectTrigger.coffee create mode 100644 app/script/connect/PhoneBook.coffee create mode 100644 app/script/conversation/ConversationMapper.coffee create mode 100644 app/script/conversation/ConversationRepository.coffee create mode 100644 app/script/conversation/ConversationService.coffee create mode 100644 app/script/conversation/ConversationStatus.coffee create mode 100644 app/script/conversation/ConversationType.coffee create mode 100644 app/script/conversation/ConversationUnreadType.coffee create mode 100644 app/script/conversation/ConversationUpdateType.coffee create mode 100644 app/script/conversation/EventMapper.coffee create mode 100644 app/script/cryptography/CryptographyError.coffee create mode 100644 app/script/cryptography/CryptographyErrorType.coffee create mode 100644 app/script/cryptography/CryptographyMapper.coffee create mode 100644 app/script/cryptography/CryptographyRepository.coffee create mode 100644 app/script/cryptography/CryptographyService.coffee create mode 100644 app/script/entity/Connection.coffee create mode 100644 app/script/entity/Conversation.coffee create mode 100644 app/script/entity/User.coffee create mode 100644 app/script/entity/message/Asset.coffee create mode 100644 app/script/entity/message/CallMessage.coffee create mode 100644 app/script/entity/message/ContentMessage.coffee create mode 100644 app/script/entity/message/DecryptErrorMessage.coffee create mode 100644 app/script/entity/message/File.coffee create mode 100644 app/script/entity/message/LinkPreview.coffee create mode 100644 app/script/entity/message/Location.coffee create mode 100644 app/script/entity/message/MediumImage.coffee create mode 100644 app/script/entity/message/MemberMessage.coffee create mode 100644 app/script/entity/message/Message.coffee create mode 100644 app/script/entity/message/NewDeviceMessage.coffee create mode 100644 app/script/entity/message/PingMessage.coffee create mode 100644 app/script/entity/message/PreviewImage.coffee create mode 100644 app/script/entity/message/RenameMessage.coffee create mode 100644 app/script/entity/message/SystemMessage.coffee create mode 100644 app/script/entity/message/Text.coffee create mode 100644 app/script/event/Backend.coffee create mode 100644 app/script/event/Client.coffee create mode 100644 app/script/event/EventError.coffee create mode 100644 app/script/event/EventRepository.coffee create mode 100644 app/script/event/EventTypeHandling.coffee create mode 100644 app/script/event/NotificationService.coffee create mode 100644 app/script/event/WebApp.coffee create mode 100644 app/script/event/WebSocketService.coffee create mode 100644 app/script/extension/GiphyContentSizes.coffee create mode 100644 app/script/extension/GiphyRepository.coffee create mode 100644 app/script/extension/GiphyService.coffee create mode 100644 app/script/links/LinkPreviewError.coffee create mode 100644 app/script/links/LinkPreviewHelpers.coffee create mode 100644 app/script/links/LinkPreviewProtoBuilder.coffee create mode 100644 app/script/links/LinkPreviewRepository.coffee create mode 100644 app/script/localization/Localizer.coffee create mode 100755 app/script/localization/strings-de.coffee create mode 100644 app/script/localization/strings-init.coffee create mode 100644 app/script/localization/strings.coffee create mode 100644 app/script/location/GeoLocation.coffee create mode 100644 app/script/main/app.coffee create mode 100644 app/script/main/auth.coffee create mode 100644 app/script/main/auth_init_dist.coffee create mode 100644 app/script/main/auth_init_edge.coffee create mode 100644 app/script/main/auth_init_prod.coffee create mode 100644 app/script/main/auth_init_staging.coffee create mode 100644 app/script/media/MediaEmbeds.coffee create mode 100644 app/script/media/MediaParser.coffee create mode 100644 app/script/message/CallMessageType.coffee create mode 100644 app/script/message/PingMessageType.coffee create mode 100644 app/script/message/SuperType.coffee create mode 100644 app/script/message/SystemMessageType.coffee create mode 100644 app/script/search/SearchLevel.coffee create mode 100644 app/script/search/SearchMode.coffee create mode 100644 app/script/search/SearchRepository.coffee create mode 100644 app/script/search/SearchResultMapper.coffee create mode 100644 app/script/search/SearchService.coffee create mode 100644 app/script/service/BackendClientError.coffee create mode 100644 app/script/service/BackendEnvironment.coffee create mode 100644 app/script/service/Client.coffee create mode 100644 app/script/storage/SkipError.coffee create mode 100644 app/script/storage/StorageKey.coffee create mode 100644 app/script/storage/StorageRepository.coffee create mode 100644 app/script/storage/StorageService.coffee create mode 100644 app/script/system_notification/SystemNotificationRepository.coffee create mode 100644 app/script/telemetry/app_init/AppInitStatistics.coffee create mode 100644 app/script/telemetry/app_init/AppInitStatisticsValue.coffee create mode 100644 app/script/telemetry/app_init/AppInitTelemetry.coffee create mode 100644 app/script/telemetry/app_init/AppInitTimings.coffee create mode 100644 app/script/telemetry/app_init/AppInitTimingsStep.coffee create mode 100644 app/script/telemetry/calling/AudioStreamStats.coffee create mode 100644 app/script/telemetry/calling/CallSetupSteps.coffee create mode 100644 app/script/telemetry/calling/CallSetupStepsOrder.coffee create mode 100644 app/script/telemetry/calling/CallSetupTimings.coffee create mode 100644 app/script/telemetry/calling/CallTelemetry.coffee create mode 100644 app/script/telemetry/calling/ConnectionStats.coffee create mode 100644 app/script/telemetry/calling/FlowTelemetry.coffee create mode 100644 app/script/telemetry/calling/MediaStreamStats.coffee create mode 100644 app/script/telemetry/calling/Stats.coffee create mode 100644 app/script/telemetry/calling/StreamStats.coffee create mode 100644 app/script/telemetry/calling/VideoStreamStats.coffee create mode 100644 app/script/tracking/EventName.coffee create mode 100644 app/script/tracking/EventTrackingRepository.coffee create mode 100644 app/script/tracking/SessionEventName.coffee create mode 100644 app/script/tracking/event/PhoneVerification.coffee create mode 100644 app/script/tracking/event/PictureTakenEvent.coffee create mode 100644 app/script/ui/Shortcut.coffee create mode 100644 app/script/ui/WindowHandler.coffee create mode 100644 app/script/user/ConnectionLevel.coffee create mode 100644 app/script/user/ConnectionStatus.coffee create mode 100644 app/script/user/UserConnectionMapper.coffee create mode 100644 app/script/user/UserError.coffee create mode 100644 app/script/user/UserMapper.coffee create mode 100644 app/script/user/UserProperties.coffee create mode 100644 app/script/user/UserRepository.coffee create mode 100644 app/script/user/UserService.coffee create mode 100644 app/script/util/ArrayUtil.coffee create mode 100644 app/script/util/BrowserPermissionType.coffee create mode 100644 app/script/util/Comparison.coffee create mode 100644 app/script/util/CountryCodes.coffee create mode 100644 app/script/util/Crypto.coffee create mode 100644 app/script/util/DebugUtil.coffee create mode 100644 app/script/util/Environment.coffee create mode 100644 app/script/util/Invite.coffee create mode 100644 app/script/util/Statistics.coffee create mode 100644 app/script/util/Time.coffee create mode 100644 app/script/util/TimeTracker.coffee create mode 100644 app/script/util/confirm.coffee create mode 100644 app/script/util/keycode.coffee create mode 100644 app/script/util/protobuf.coffee create mode 100644 app/script/util/scroll-helpers.coffee create mode 100644 app/script/util/shims.coffee create mode 100644 app/script/util/storage.coffee create mode 100644 app/script/util/types.coffee create mode 100644 app/script/util/util.coffee create mode 100644 app/script/view_model/ActionsViewModel.coffee create mode 100644 app/script/view_model/ArchiveViewModel.coffee create mode 100644 app/script/view_model/AuthViewModel.coffee create mode 100644 app/script/view_model/BackgroundViewModel.coffee create mode 100644 app/script/view_model/CallShortcutsViewModel.coffee create mode 100644 app/script/view_model/ConnectRequestsViewModel.coffee create mode 100644 app/script/view_model/ConversationInputViewModel.coffee create mode 100644 app/script/view_model/ConversationListViewModel.coffee create mode 100644 app/script/view_model/ConversationTitlebarViewModel.coffee create mode 100644 app/script/view_model/DebugViewModel.coffee create mode 100644 app/script/view_model/GiphyViewModel.coffee create mode 100644 app/script/view_model/ImageDetailViewViewModel.coffee create mode 100644 app/script/view_model/LoadingViewModel.coffee create mode 100644 app/script/view_model/MainViewModel.coffee create mode 100644 app/script/view_model/MessageListViewModel.coffee create mode 100644 app/script/view_model/ModalsViewModel.coffee create mode 100644 app/script/view_model/ParticipantsViewModel.coffee create mode 100644 app/script/view_model/RightViewModel.coffee create mode 100644 app/script/view_model/SelfProfileViewModel.coffee create mode 100644 app/script/view_model/SettingsViewModel.coffee create mode 100644 app/script/view_model/StartUIViewModel.coffee create mode 100644 app/script/view_model/VideoCallingViewModel.coffee create mode 100644 app/script/view_model/WarningsViewModel.coffee create mode 100644 app/script/view_model/WelcomeViewModel.coffee create mode 100644 app/script/view_model/WindowTitleViewModel.coffee create mode 100644 app/script/view_model/bindings/CommonBindings.coffee create mode 100644 app/script/view_model/bindings/ConversationListBindings.coffee create mode 100644 app/script/view_model/bindings/ListBackgroundBindings.coffee create mode 100644 app/script/view_model/bindings/MessageListBindings.coffee create mode 100644 app/script/view_model/bindings/SearchBindings.coffee create mode 100644 app/style/auth/account.less create mode 100644 app/style/auth/animations.less create mode 100644 app/style/auth/auth.less create mode 100644 app/style/auth/base.less create mode 100644 app/style/auth/limit.less create mode 100644 app/style/auth/posted.less create mode 100644 app/style/auth/verify.less create mode 100644 app/style/base.less create mode 100644 app/style/common/animations.less create mode 100644 app/style/common/bubble.less create mode 100644 app/style/common/button-round.less create mode 100644 app/style/common/button.less create mode 100644 app/style/common/checkbox.less create mode 100644 app/style/common/colors.less create mode 100644 app/style/common/common.less create mode 100644 app/style/common/hide-controls.less create mode 100644 app/style/common/hint.less create mode 100644 app/style/common/input.less create mode 100644 app/style/common/label.less create mode 100644 app/style/common/link.less create mode 100644 app/style/common/mixins.less create mode 100644 app/style/common/modal.less create mode 100644 app/style/common/shapes.less create mode 100644 app/style/common/slider.less create mode 100644 app/style/common/spinner.less create mode 100644 app/style/common/three-dots.less create mode 100644 app/style/common/variables.less create mode 100644 app/style/components/accent-color-picker.less create mode 100644 app/style/components/asset/audio-asset.less create mode 100644 app/style/components/asset/common/common.less create mode 100644 app/style/components/asset/controls/audio-seek-bar.less create mode 100644 app/style/components/asset/controls/media-button.less create mode 100644 app/style/components/asset/controls/seek-bar.less create mode 100644 app/style/components/asset/file-asset.less create mode 100644 app/style/components/asset/link-preview-asset.less create mode 100644 app/style/components/asset/location-asset.less create mode 100644 app/style/components/asset/video-asset.less create mode 100644 app/style/components/calling/choose-screen.less create mode 100644 app/style/components/calling/device-toggle-button.less create mode 100644 app/style/components/common-contacts.less create mode 100644 app/style/components/device-card.less create mode 100644 app/style/components/device-remove.less create mode 100644 app/style/components/group-list.less create mode 100644 app/style/components/input-element.less create mode 100644 app/style/components/search-list.less create mode 100644 app/style/components/user-avatar.less create mode 100644 app/style/components/user-input.less create mode 100644 app/style/components/user-list.less create mode 100644 app/style/components/user-profile.less create mode 100644 app/style/conversation/confirm.less create mode 100644 app/style/conversation/connect-requests.less create mode 100644 app/style/conversation/conversation-input.less create mode 100644 app/style/conversation/conversation-titlebar.less create mode 100644 app/style/conversation/conversation.less create mode 100644 app/style/conversation/detail-view.less create mode 100644 app/style/conversation/giphy.less create mode 100644 app/style/conversation/message-list.less create mode 100644 app/style/conversation/participants.less create mode 100644 app/style/debug.less create mode 100644 app/style/fonts/README.md create mode 100755 app/style/fonts/zeta-neue.css create mode 100644 app/style/framework.less create mode 100644 app/style/list/actions.less create mode 100644 app/style/list/archive.less create mode 100644 app/style/list/background.less create mode 100644 app/style/list/conversation-list-call.less create mode 100644 app/style/list/conversation-list.less create mode 100644 app/style/list/invite-bubble.less create mode 100644 app/style/list/start-ui.less create mode 100644 app/style/loading-screen.less create mode 100644 app/style/main.less create mode 100644 app/style/self/self-about.less create mode 100644 app/style/self/self-profile.less create mode 100644 app/style/self/self-settings.less create mode 100644 app/style/support.less create mode 100644 app/style/video-calling.less create mode 100644 app/style/warnings.less create mode 100644 app/style/welcome.less create mode 100644 aws/application.py create mode 100644 aws/config.py create mode 100644 aws/libs/__init__.py create mode 100644 aws/libs/flask_sslify.py create mode 100644 aws/libs/httpagentparser/__init__.py create mode 100644 aws/libs/httpagentparser/more.py create mode 100644 aws/main.py create mode 100644 aws/requirements.txt create mode 100644 aws/templates/robots.txt create mode 100644 aws/templates/sitemap.xml create mode 100644 aws/util.py create mode 100644 bin/country_codes.py create mode 100644 bower.json create mode 100644 coffeelint.json create mode 100644 crowdin.yaml create mode 100644 grunt/config/aws_s3.coffee create mode 100644 grunt/config/bower.coffee create mode 100644 grunt/config/clean.coffee create mode 100644 grunt/config/codo.coffee create mode 100644 grunt/config/coffee.coffee create mode 100644 grunt/config/coffeelint.coffee create mode 100644 grunt/config/compress.coffee create mode 100644 grunt/config/connect.coffee create mode 100644 grunt/config/copy.coffee create mode 100644 grunt/config/includereplace.coffee create mode 100644 grunt/config/karma.coffee create mode 100644 grunt/config/less.coffee create mode 100644 grunt/config/open.coffee create mode 100644 grunt/config/postcss.coffee create mode 100644 grunt/config/shell.coffee create mode 100644 grunt/config/todo.coffee create mode 100644 grunt/config/uglify.coffee create mode 100644 grunt/config/watch.coffee create mode 100644 grunt/tasks/aws.coffee create mode 100644 grunt/tasks/misc.coffee create mode 100644 grunt/tasks/raygun.coffee create mode 100644 grunt/tasks/scripts.coffee create mode 100644 karma.conf.js create mode 100644 package.json create mode 100644 test/api/OpenGraphMocks.js create mode 100644 test/api/SDP_payloads.js create mode 100644 test/api/TestFactory.js create mode 100644 test/api/calling-payloads/call.flow-active.md create mode 100644 test/api/calling-payloads/call.flow-add.md create mode 100644 test/api/calling-payloads/call.flow-delete.md create mode 100644 test/api/calling-payloads/call.remote-sdp.md create mode 100644 test/api/calling-payloads/call.state.md create mode 100644 test/api/calling-payloads/conversation.voice-channel-activate.md create mode 100644 test/api/calling-payloads/conversation.voice-channel-deactivate.md create mode 100644 test/api/environment.js create mode 100644 test/api/payloads.js create mode 100644 test/unit_tests/announce/AnnounceRepositorySpec.coffee create mode 100644 test/unit_tests/announce/AnnounceServiceSpec.coffee create mode 100644 test/unit_tests/assets/AssetCryptoSpec.coffee create mode 100644 test/unit_tests/assets/AssetRemoteDataSpec.coffee create mode 100644 test/unit_tests/auth/AuthRepositorySpec.coffee create mode 100644 test/unit_tests/cache/CacheRepositorySpec.coffee create mode 100644 test/unit_tests/calling/CallCenterSpec.coffee create mode 100644 test/unit_tests/calling/CallRequestResponseMock.coffee create mode 100644 test/unit_tests/calling/entities/FlowSpec.coffee create mode 100644 test/unit_tests/calling/mapper/SDPMapperSpec.coffee create mode 100644 test/unit_tests/client/ClientMapperSpec.coffee create mode 100644 test/unit_tests/client/ClientRepositorySpec.coffee create mode 100644 test/unit_tests/client/ClientSpec.coffee create mode 100644 test/unit_tests/connect/ConnectGoogleServiceSpec.coffee create mode 100644 test/unit_tests/conversation/ConversationMapperSpec.coffee create mode 100644 test/unit_tests/conversation/ConversationRepositorySpec.coffee create mode 100644 test/unit_tests/conversation/ConversationServiceSpec.coffee create mode 100644 test/unit_tests/conversation/EventMapperSpec.coffee create mode 100644 test/unit_tests/cryptography/CryptographyMapperSpec.coffee create mode 100644 test/unit_tests/cryptography/CryptographyRepositorySpec.coffee create mode 100644 test/unit_tests/entity/ConversationSpec.coffee create mode 100644 test/unit_tests/entity/UserSpec.coffee create mode 100644 test/unit_tests/entity/message/FileSpec.coffee create mode 100644 test/unit_tests/entity/message/MediumImageSpec.coffee create mode 100644 test/unit_tests/entity/message/MemberMessageSpec.coffee create mode 100644 test/unit_tests/entity/message/MessageEntitiesSpec.coffee create mode 100644 test/unit_tests/event/EventRepositorySpec.coffee create mode 100644 test/unit_tests/extension/GiphyRepositorySpec.coffee create mode 100644 test/unit_tests/links/LinkPreviewHelpersSpec.coffee create mode 100644 test/unit_tests/links/LinkPreviewProtoBuilderSpec.coffee create mode 100644 test/unit_tests/links/LinkPreviewRepositorySpec.coffee create mode 100644 test/unit_tests/localization/LocalizerSpec.coffee create mode 100644 test/unit_tests/location/GeoLocationSpec.coffee create mode 100644 test/unit_tests/media/MediaEmbedsSpec.coffee create mode 100644 test/unit_tests/search/SearchRepositorySpec.coffee create mode 100644 test/unit_tests/service/ClientSpec.coffee create mode 100644 test/unit_tests/system_notification/SystemNotificationRepositorySpec.coffee create mode 100644 test/unit_tests/tracking/EventTrackingRepositorySpec.coffee create mode 100644 test/unit_tests/ui/ShortcutSpec.coffee create mode 100644 test/unit_tests/user/UserMapperSpec.coffee create mode 100644 test/unit_tests/user/UserRepositorySpec.coffee create mode 100644 test/unit_tests/user/UserServiceSpec.coffee create mode 100644 test/unit_tests/util/ArrayUtilSpec.coffee create mode 100644 test/unit_tests/util/ComparisonSpec.coffee create mode 100644 test/unit_tests/util/CountryCodesSpec.coffee create mode 100644 test/unit_tests/util/CryptoSpec.coffee create mode 100644 test/unit_tests/util/DebugUtilSpec.coffee create mode 100644 test/unit_tests/util/EnvironmentSpec.coffee create mode 100644 test/unit_tests/util/InviteSpec.coffee create mode 100644 test/unit_tests/util/TimeSpec.coffee create mode 100644 test/unit_tests/util/UtilSpec.coffee create mode 100644 test/unit_tests/util/bindingsSpec.coffee create mode 100755 trans.py diff --git a/.bowerrc b/.bowerrc new file mode 100644 index 00000000000..cf4ec5b9cd1 --- /dev/null +++ b/.bowerrc @@ -0,0 +1,3 @@ +{ + "directory": "temp/bower_components" +} diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 00000000000..9d08a1a828a --- /dev/null +++ b/.editorconfig @@ -0,0 +1,9 @@ +root = true + +[*] +charset = utf-8 +indent_style = space +indent_size = 2 +end_of_line = lf +insert_final_newline = true +trim_trailing_whitespace = true diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 00000000000..34d849c83db --- /dev/null +++ b/.gitattributes @@ -0,0 +1,15 @@ +# Reference: https://help.github.com/articles/dealing-with-line-endings/ + +# enforce LF line endings +* text eol=lf + +# Denote all files that are truly binary and should not be modified. +*.eot binary +*.jar binary +*.jpg binary +*.otf binary +*.png binary +*.svg binary +*.ttf binary +*.woff binary +*.woff2 binary diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000000..cf3bfaf164d --- /dev/null +++ b/.gitignore @@ -0,0 +1,38 @@ +*.pyc +.DS_Store +.grunt +.Icon +/_SpecRunner.html +/app/ext +/aws/s3 +/aws/static +/aws/templates/**/*.html +/aws/templates/auth +/aws/version +/deploy +/dist +/docs +/nbproject +/node_modules +/temp +/test/_SpecRunner.html +/test/coverage +/test/js/* +/Web.config +aws-keys.json +Gruntfile.js +npm-debug.log +strings-ar.coffee +strings-fr.coffee +strings-it.coffee +strings-pt.coffee +strings-ru.coffee + +# Automatically created files +/doc + +# IntelliJ +# http://devnet.jetbrains.com/docs/DOC-1186 +/.idea/**/*.xml +!/.idea/codeStyleSettings.xml +!/.idea/modules.xml diff --git a/.idea/codeStyleSettings.xml b/.idea/codeStyleSettings.xml new file mode 100644 index 00000000000..fbfe254af26 --- /dev/null +++ b/.idea/codeStyleSettings.xml @@ -0,0 +1,46 @@ + + + + + + \ No newline at end of file diff --git a/.idea/modules.xml b/.idea/modules.xml new file mode 100644 index 00000000000..468998ad51a --- /dev/null +++ b/.idea/modules.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/wire-webapp.iml b/.idea/wire-webapp.iml new file mode 100644 index 00000000000..b0452cd8fbe --- /dev/null +++ b/.idea/wire-webapp.iml @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.travis.yml b/.travis.yml new file mode 100644 index 00000000000..07daeb53d95 --- /dev/null +++ b/.travis.yml @@ -0,0 +1,42 @@ +# http://docs.travis-ci.com/user/workers/container-based-infrastructure/ +sudo: required +dist: trusty + +# https://docs.travis-ci.com/user/installing-dependencies/#Installing-Packages-with-the-APT-Addon +addons: + apt: + sources: + - google-chrome + packages: + - google-chrome-stable + +# http://docs.travis-ci.com/user/languages/javascript-with-nodejs/ +language: node_js +node_js: + - node + +# https://blog.travis-ci.com/2013-12-05-speed-up-your-builds-cache-your-dependencies +cache: + directories: + - bower_components + - node_modules + +env: + global: + # RAYGUN_USERNAME + - secure: "xMiUg9p7HCDUarPgsb5CVuTTnYWAZfJf1a8AJA0WKLg1XY3hN/CayYE4jxor+hW2GZVhAeUFPHJXo0OpF9yhXdspVG4i6p/ZAq97N2rlGIHXngTGp6J3pCRtC1wIJqG9g45lST84IfrP0yqnMcY2rnQkE0ZbY9Qn4gIvCEjXJpzA8aI4Lqr/K/LI3cYdnlgPu0s9E2I/Kl0ZenDqD9W0TFyufWNXzH8HqyrEw7UjRM9d8oosYydM5bNmW4oCrMrLMDcqbwC8PPvJUrJKt8tubDuqDDNVECiSE2Rz8FPA3r017heQOOGinzAiUkMVHlJEdcPxBViPhGE6BN10QC1vR6uxaM/9UBIUodLfculzhZ5xcHYgEVHC1w4M5+zoa4F8iecUMMwT1+qC6Na/apTpUNy+OLguK4Se7dTKtfd4+FTbWZANI8xT5duVNnJ//rH6U06OpXDKNvWlYyGl2zZr0gcTXGqxKU+7kVjNZa1k6ltOgy86pTm4VAOLBGfqUbSwRVhW3aqJKS/yjsAkT9oLKuB3T0osVPcsSxS4MmODUmUazr7UMZr8t2S0L0v0NE/Q1I+gykqEzV8ecd9s6RXaWQiwZXgSjMe3UR3qrFO+r1fbwXIo45biPyd1rDFR1Y/dPHxDCPOV6fXjWXp68r5d9tcZTc6xjU1vgoplJfxAyhA=" + # RAYGUN_PASSWORD + - secure: "29BtoWU0D68HVLqJ5iG3QNs+B2ytaZsTRsgW5T0WBCWS8b4cOOFLjwQHh5f+Qr0oPjU+cwQ0Sequ9wzGK882lhoT5SyddURhnj2CxWko0zmeNst2YpjEEdrgvtqQMGG+SXizjG9gcPcf52mF0du7LdICh1pN5HZMVckPD02t4+quZn0SpwdvQSCdWtivSNSenguhdJg9z6ulKKJUV3nqvdJsSAC48q9awEfc59dvo8nf7v44bTlIcqGkiCMpR2pxPXjwOvogWQcUZvd2onSZvtOJWOega5iAc0PvSs9p2eLpY82lQSlvsLxaytSiQw50rDLYi1p4ouUn//DUH0Ir8A3g57MRzXkMLs1JDXOJyTbPgLkOdB/TmlryQ2iT0XnHp4ncjWrWjmgQPDKizk7HFeSvMPKjo9V2F46FwXClq1b/U1jP5PVOQHwtujVVmp9CXvBedz1SLaY2GIe0uLexFis1P5UZSCmtP/GpnO47JIguYU43Xkre2hdn1pnpCHiDv4xWISfinBZa+gncjAEodKkApysx48cK6nbZ8HLPnMGDk04oZO+zFz2IprkkefdoKJ32J6BnBtQqmJAXwPaC+AiIEcHWTbHRtZM0NvaxsQFB/MPCy+A2H1amC63wE/CJZbGf2sNQZCNUwsFq8awRxHt8frSS/9Rw+Wg6jDMZp+w=" + +# http://docs.travis-ci.com/user/gui-and-headless-browsers +before_install: + - export CHROME_BIN=/usr/bin/google-chrome + - export DISPLAY=:99.0 + - sh -e /etc/init.d/xvfb start + +# http://docs.travis-ci.com/user/pull-requests/ +script: + - npm test + +notifications: + email: false diff --git a/Gruntfile.coffee b/Gruntfile.coffee new file mode 100755 index 00000000000..7e618d91d49 --- /dev/null +++ b/Gruntfile.coffee @@ -0,0 +1,146 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +# @formatter:off +module.exports = (grunt) -> + require('load-grunt-tasks') grunt + + path = require 'path' + + config = + aws: + port: 5000 + server: + port: 8888 + + dir = + app_: 'app' + app: + demo: 'app/demo' + ext: 'app/ext' + page: 'app/page' + script: 'app/script' + style: 'app/style' + template_dist: 'app/page/template/_dist' + aws_: 'aws' + aws: + s3: 'aws/s3' + static: 'aws/static' + templates: 'aws/templates' + deploy: 'deploy' + dist: 'dist' + docs: + api: 'docs/api' + coverage: 'docs/coverage' + temp: 'temp' + test_: 'test' + test: + api: 'test/api' + coffee: 'test/coffee' + coverage: 'test/coverage' + js: 'test/js' + lib: 'test/lib' + unit_tests: 'test/unit_tests' + + grunt.initConfig + pkg: grunt.file.readJSON 'package.json' + config: config + dir: dir + aws_s3: require "./grunt/config/aws_s3" + bower: require "./grunt/config/bower" + clean: require "./grunt/config/clean" + codo: require './grunt/config/codo' + coffee: require "./grunt/config/coffee" + coffeelint: require "./grunt/config/coffeelint" + compress: require "./grunt/config/compress" + connect: require "./grunt/config/connect" + copy: require "./grunt/config/copy" + includereplace: require "./grunt/config/includereplace" + karma: require "./grunt/config/karma" + less: require "./grunt/config/less" + open: require "./grunt/config/open" + path: require "path" + shell: require "./grunt/config/shell" + todo: require "./grunt/config/todo" + uglify: require "./grunt/config/uglify" + watch: require "./grunt/config/watch" + postcss: require "./grunt/config/postcss" + +############################################################################### +# Tasks +############################################################################### + grunt.loadTasks 'grunt/tasks' + grunt.registerTask 'default', ['prepare_dist', 'host'] + grunt.registerTask 'init', ['clean:ext', 'clean:temp', 'bower', 'scripts'] + +############################################################################### +# Deploy to different environments +############################################################################### + grunt.registerTask 'app_deploy', ['gitinfo', 'aws_deploy'] + grunt.registerTask 'app_deploy_edge', ['gitinfo', 'set_version:edge', 'aws_deploy'] + grunt.registerTask 'app_deploy_staging', ['gitinfo', 'set_version:staging', 'aws_deploy'] + grunt.registerTask 'app_deploy_prod', ['gitinfo', 'set_version:prod', 'aws_deploy'] + grunt.registerTask 'app_deploy_taco', ['gitinfo', 'set_version:taco', 'aws_deploy'] + + grunt.registerTask 'app_deploy_travis', (target) -> + if target in ['prod', 'staging'] + grunt.task.run "set_version:#{target}", 'init', "prepare_#{target}", 'aws_prepare' + else if target is 'dev' + grunt.task.run "set_version:staging", 'init', "prepare_staging", 'aws_prepare' + else + grunt.fail.warn 'Invalid target specified. Valid targets are: "prod" & "staging"' + +############################################################################### +# Test Related +############################################################################### + grunt.registerTask 'test', -> + grunt.task.run ['clean:docs_coverage', 'scripts', 'test_init', 'test_prepare', 'karma:test'] + + grunt.registerTask 'test_prepare', (test_name) -> + scripts = grunt.config 'scripts' + # Little hack because of a configuration bug in "grunt-karma": + # @see https://github.com/karma-runner/grunt-karma/issues/21#issuecomment-27518692 + prepare_file_names = (file_name_array) => + return (file_name.replace 'deploy/', '' for file_name in file_name_array) + + helper_files = grunt.config.get 'karma.options.files' + app_files = prepare_file_names scripts.app + component_files = prepare_file_names scripts.component + vendor_files = prepare_file_names scripts.vendor + test_files = if test_name then ["../test/js/#{test_name}Spec.js"] else ['../test/**/*Spec.js'] + + files = [].concat helper_files, vendor_files, component_files, app_files, test_files + grunt.config 'karma.options.files', files + + grunt.registerTask 'test_init', ['prepare_dist', 'prepare_test'] + + grunt.registerTask 'test_run', (test_name) -> + grunt.config 'karma.options.reporters', ['progress'] + grunt.task.run ['scripts', 'newer:coffee:dist', 'newer:coffee:test', "test_prepare:#{test_name}", 'karma:test'] + +############################################################################### +# Documentation +############################################################################### + grunt.registerTask 'generate_docs', (command) -> + switch command + when 'all' + grunt.task.run ['clean:docs', 'codo:app', 'test:coverage'] + else + grunt.task.run ['clean:docs', 'codo:app'] + +# @formatter:on diff --git a/LICENSE b/LICENSE new file mode 100644 index 00000000000..9cecc1d4669 --- /dev/null +++ b/LICENSE @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + {one line to give the program's name and a brief idea of what it does.} + Copyright (C) {year} {name of author} + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + {project} Copyright (C) {year} {fullname} + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/README.md b/README.md new file mode 100644 index 00000000000..a7d059a7f9e --- /dev/null +++ b/README.md @@ -0,0 +1,48 @@ +# Wire™ + +![Wire logo](https://github.com/wireapp/wire/blob/master/assets/logo.png?raw=true) + +This repository is part of the source code of Wire. You can find more information at [wire.com](https://wire.com) or by contacting opensource@wire.com. + +You can find the published source code at [github.com/wireapp/wire](https://github.com/wireapp/wire). + +For licensing information, see the attached LICENSE file and the list of third-party licenses at [wire.com/legal/licenses/](https://wire.com/legal/licenses/). + +If you compile the open source software that we make available from time to time to develop your own mobile, desktop or web application, and cause that application to connect to our servers for any purposes, we refer to that resulting application as an “Open Source App”. All Open Source Apps are subject to, and may only be used and/or commercialized in accordance with, the Terms of Use applicable to the Wire Application, which can be found at https://wire.com/legal/#terms. Additionally, if you choose to build an Open Source App, certain restrictions apply, as follows: + +a. You agree not to change the way the Open Source App connects and interacts with our servers; b. You agree not to weaken any of the security features of the Open Source App; c. You agree not to use our servers to store data for purposes other than the intended and original functionality of the Open Source App; d. You acknowledge that you are solely responsible for any and all updates to your Open Source App. + +For clarity, if you compile the open source software that we make available from time to time to develop your own mobile, desktop or web application, and do not cause that application to connect to our servers for any purposes, then that application will not be deemed an Open Source App and the foregoing will not apply to that application. + +No license is granted to the Wire trademark and its associated logos, all of which will continue to be owned exclusively by Wire Swiss GmbH. Any use of the Wire trademark and/or its associated logos is expressly prohibited without the express prior written consent of Wire Swiss GmbH. + +# How to build the open source client + +### Requirements + +- Install [Node.js](https://nodejs.org/) +- Install [Grunt](http://gruntjs.com/): `npm install -g grunt-cli` + +### Run Wire for Web locally + +The first time you run the project you should install the dependencies by +executing the following from your terminal: + +```bash +npm install +``` + +To run the actual server: + +```bash +grunt +``` + +If everything went well the app will be available on +[`localhost:8888`](http://localhost:8888). + +### Generate code coverage + +```bash +grunt test:coverage +``` diff --git a/app/audio/buzzer/applause.mp3 b/app/audio/buzzer/applause.mp3 new file mode 100644 index 0000000000000000000000000000000000000000..3348bee3af2e08b4f229c868f3a0083ac3248c51 GIT binary patch literal 84038 zcmXtO&o2@H0Ht8!Zo$RI%*M{l4t}i%0pLPNyamz7&74(G(oO&r z4l0H_@cGG8<22p7<(T*#_y80$GeVxfD<2|R2ZWGh05iN{ZU6_~UvvP1#sAEbj)QN3 zBRL~uV?Mb|N$sbFHJ?&W&y8;hi?CwPO0S$+k`JR!_vL$2izW1UHXdI+9v-b-JP;3$ z1P>1n6LlqV=EkK-K&lb<`WMs)Y0iD1S=ElN91(B-I3Bz`JOa`RBtS$$da6zUZE6I- z3os^Xg9sc5Ljk-w<~5T+iJ7I7LBIjXAi!+kd%WRMQt&@J`JoI3_k!;qXoOJ7*?+`( zXUA0Rc+IusK*FIgVO_f1=lgRMz@Y((vMGuF1j1bsor*!D`4IQRsm_h~+XIfkInnMl6(kR;So&2}~%oearY!33TynTT`T)?W?B$p4V>PSNy!ZvL2BvuX;GQ z7=P6r*y`1-*=n}A=cUxt8PPEedw6?ryV?KhGEwsySxk{90)R3TfVb{AxO}%(>pi*F=va^m})BiV(V73w<2W20Pen=em`0 zP$F6+B{jnL!!~i30hKyVCot(I4NogAAVq*^M};7~3_ze{yzU%4#X$}-P@=WTLrF1? zPxBKSCHZPA|Hnu0oBS{LtZw>q_M@Ww%2CrK+5sp8$^agx>x~w(kL&9x*@d^OH{aFl z%Q3p_w5^?_q^H(Yixwg6oAt->uYBw4|H*=18$s0m!SP;wR@5BMg*Q3~(AEx^CMQrD zSpEp|-Mf{6@ep`YZmPJ=MGsRA!33gqG|2{0NpVH5y)5en>>!!bqAwbJ0!8FFXT{> zR0TBlx0g>h7u4qr*M?MJL?y-zmy?m8toOPTi~G&QDye z>(Pf{PTWT-{ahDXnmy)*B93=sHjcq2+)HxD5u;N>KszjJUTHo>Z1nU@Hu%RtSv-X~ z(G(TFLQgD>49KHdMYQ@1d_2$`#lE%bT&l#nE1|O(6dr(Vp|#bY#3dBnXXfO7hN&lP zP%{UcMbPx9QJT|9oK7}R^6oJKvx0vH`QxN_yuf*_?0~6yh?1+-jV1S@0PZQ1*o0y-ASi5s7H(q0f$$ZxhCwa0Dj<^Yo3qqy+*FUT+SLMF)AU)m{++q9C zziE{ZYhovw9&w!uog1EghZpmq8PiuZ%lJ7;gq|Lprk;NORizxAJ zag@S%%>T7Lzu%sQ=~y|VPpm%)C|+Ce_95)Ib93_y27kZ08ha5jbb{@9Y+w=N3UJ$f zeTZ|ho1k8FNzy)@_q|07PE#Q=`{Rw!D$X z5H0BM;0H5bXZs;toD11U{p%A&pM{;Lijq+4J#JQAd5BcZAUo>JofoyN53u?=wY6o( zFwrt~aD>>z=9f@em#2I1%cnB=_tq)|5=Y zg2vH%>&S|Uf8B}lLjV|J772dP)ARZe=1#lCBk(227ODwl|Cg`x)0`wdq94mrmfbCH zAG(#ULCXF$Ohw5DHEGAGEBKDSt=x^O5&`ryeGhheZ28h3Najm|4*$ExT{Wt?gZrw% z3}S#S5Bz8cF70XibfUMcxP4w57dna!jRxC`3uimtT;zmi4lbyj015I0O5saCNF@?5 zI~67$xEl6ztMv{eWj09Gl@%0 zoqz;=a+q>RtX3Iu6iF}0`~ndZdT?Bv%B2<)k{A+Wv$hN5I7BqWTwm=#2{+dc#-mnD zxS~nm#U&NP^|O{Q;72zGgDo)lrNj^X!pH@Fr1wVy019Q5zI((}Ml~^DgN{0Y4rv&h zlGMRVNHqCQq_r&gi=(boWvPW2>$e7n|B%D=hhei={w}l&8ByWgb5ak`h$xaT+nwqI zQ_^)mWjHP2KW^`1S*bJ_JXaaC4sI2ESCi$`Dxa6fx!v3U&WP9g{y_Keg5zeh{@*73 z-OTbhnChQYy#ZHi^ZoMK7I)Y!a$q@lj$nZ=5(t8@qbkd^iKAv z!&<3XZfv(pmKWVu7M9Ly0vR{H(~}<#-&$Mhep()WJl!{gVDLTUoTKHB>*%&Wc0P>2 z>|0}ZpN@Q5nov5SdU5q5KVm$rt~G>C%=j+lMv>oZOciv(iU~ybDK?f?^so-mL-C$CpSu(LXeG2T{Itsf?WBMP5JpV zZFZu<(2F0$f{q(i`s9-6gDeI6xj%5b=?buk8L;cYXY}5N+Ya4BRKq4LAAP7Ep~TRX44+Sk6aud!ey^hf2XH&9FiZ> zws-`W60~DzCE+I)jxUB+>0Snde1{+3_`bnci-vL`}Gs7Ih~n(g^7YMW;23 z8{sG;>536tE0ST2*!ri3(HU3fz4ae**k54OE#_4J$~*gvt@8^V8)3E-30?{z_q`EW zKKw`JWYPRFVU~E|)Up1sY{_s}tk-&^(fh>bJS7M3?$MI`Anid*hChdEpNdvMMnh$e zaLGC^V682tQB&yG8No@mIHtQc0hcr;TetK_nopi&`ES0|h`cI4sB#2PWnGy+&G~N( z!iM#53ya#%KBn>1tKn9@G@|j`keEP`3lH&Mo z?KyW13^&u{U9PNT$IU$9Vk_kmKf7%l(~j@@({9;PH8$JvCM7A#3!=X9bvjJ81vggC=%9=v$Ay>lU zS4?Vg4?;JN$o19Dv`Xfj<;G{m&1Qz;`r0skainuo`x4;{{Vmdgd!ZoW_@V;l&Z+9v{pX^y7Niar);3V{T^c`*aPZPqGUKV0J1hdq1*fA<`kGf2F{@n5pHY-uhd+84kVzEUpI!`6F zi-8u?CmM&H8{T*|)!G}_TzWW)Ub2HeK`z0{A#jCv`b%JTp9lk4z zZT=)2K9lok?PZCs3|mp#lkEEHc?8n%bYdY%3(_0Q*wNy6)S(Vw#$;b(rvgbd>TMM` z&$ZwWFg{QGQmKwHl4yb`F%BN6=&iuiT~g&x%fxraHlwhr9LHJ>Wo^`&)QPMT_BxCx z(IGY9Uw>v|V)6R-Z?y6`&RF2b$XG28%+T(N2S7yU>~V%4Cy z(nyMa3uyo1d5Ve(WVuk;U)T__gfInS3*mV}SbEcilrRzn=j$a5kI7OSFEUHx`VID5 zLFe14hTFPhXUx+FuQxV(<+9acrx|z!|B$1Mh91+D&-lxJ7asmxDRgF_Y3Kx0FlXuq z(oto<;d>r}boyXb;%9> z+B;}JFvZ#i9(&Bs{l0^18pr_L9kc5ZJP}``zCd(@P9#W3lz_gOhJA^qNW#GSf-KgI zlL(dOWt@DYBeP-q4G)2f+MhIY-gDF8Q8&Xb@k&qLFM5gU#qLAb@@lmpXwOmbDs+=r z(;H}qUjO6k9O-_@2xNjfQUCLv z^F+uNaAP0)uqr>$&tHEr~ML%Pu%9;$9CZuxt0{_foprLN8ZV%L%j84DY$ z6l|l*vp&$nNi}yk0dW)Lw!Hc!uHD-@aa7Mks~jzuPg)Cfwfs2t7+5#eeDsu!-i>!< z$LB;l!?QXU*92;o|EvT63~Y>6OOWA7y^FsEP0j>5X;zFVg~opPOVHvJ;!s?xArIBo z^LUg0OcbE6((3&tSfb53;JTRVmA5kCd9VEsIWiRJX3cRzJoY~km1CAd#}3#nnq&$` zQDe8Ir<}LREcK0Tvcphc0F!uZ31a?9Ni^#X%lzFNp8i4;5I#$b!A%Vs9fKw1PPDgF zGb4;Ktk~jPrBDpKnW1H-T}kAKebE}fVKzDGw&WV);A%SPS#PxFJsz+#J=o}4Xf^%$ zjX`E&YG1{JVV&fWBr6xrsIMPlOAKxGWik!*wTMoXea&E~zEuNm>9-(YKR6f~r6^D= zi@JHSAM&>qj(>}kp4~DI{f=%0cA8P4xBi7wmzpi+dSKOnTBkl{3*Ci$sVtmwqyI_{ zd0@ts!TXERh3!P1&$c&e(1rfZy}pCh9+y}6UP`Uj3G?8C;?*jf>o)dVM!MA%aYU5# z#b#tn*Mh#Q!GAk1NuJ)n?(Z)CJ~1Wokq$q1FdwkczTD_{^)yaT(m(zF#zqTZ;A*?N zWycF-Y}XofA|793vO;C@#C^Ky8*mv|E3wGLSi{|FiX2xxshuay(Hx4ul&;FOiv6%q+GO)hSJL zsG=c$=@cmb@0!>ryOAgU!o8B39`?pI9iit>S8F5y0Ik?Y)WEdE3!OTSc8-}uVi1`8 zTcu$dz7qnC3<^+vn15~JI)hw#vA#O!{;MmYT`9`6PkNpHJO`NWD}*z^{d4?@fW^Wl z*W#n?nO3Rkn;=xzH>E||=t_+9guVh0X!+|3M$nq7xR!m&;8wTl@^P6?9 z+^ZBzSDA1&zKzJFa=Z&HIIyZqza&Vo3r?k0fA~9G_AcC}E4B$IazCdOJ z5OCGB0WcL2|92==A^g(7^pFeuI{&puT2UHZV>6otKPv{(2|DW70wfp_Yl6-Hjpbhk z{|gNM$YPqJ>=F~M`}dD3$nn%&H{fU#VP4Gkwa$b1x9mW5DwVWEP$tz?oEUzGFU^-T zJ*$Mw;7MkvGo{Vdvw#L_n&A9Dc%rO|7n86~t}jxdq2sJ98Q{{h+J-QiOpLQcYrlY7sEaN>o$Uu}!OW*ODqjVK#Zp+uAPO@j zXKW;_^NWxI+w~LRajanK>oX{ryddtwRTVYhfetyUm*B zsmFug1T5I1&5$QrRVpWK-a4Hu*i(mj@OF~B%?Mf|5-7asD-2BZTEW%k;NApa=5>vF zu_WgY>de%_=rt2cWo+3Fa!pl*^;ro(f+X-T#Rwfk>WY07*NP6)YobuKf5k?UPLw4= zl25-x@@vc5u#s143#GLif$!*uh zd+PGl)rJqZ%OX7zEKWVv_u~tRibI(s-6axq+0T=tKZ0y)$-H*LoS;$pVN^>w(lYU2 zekzqe?@xJ!DLvOIZ_nfL86oQjUBc=Sgo)}Wrc}N8C(8Ajmdb($_@o~0#mqC-g5#tp zozD1nl8r1(<}OE(%HyqjW}~lm)b%lH%Yv2Cf0#2=Jpocr8<57rAOfF3zK)m%;T0gkgKT>Ztq`PRH z7I{51DA(4ikHwo}RVH%BeU@{;^D{j%qo48f?mO&QXv#B0wMA0ow-?Z3r}VAsNL|X9 z1qUV|S)kd3vIF z30Yeu(Gb`@6lN32G}f%<4>Kx`i8^X%;LNTpV|f8`!!5O@6-Dgq>=Z-)!z2!)4HvH5Y`#3#1^EF}eE}QR+l;VY zscD(Jv)_t$1m=}4Y^`lSTw3!;TF-L7y|-m+bxh}QCs`xl)1%o$cCMrs3SMv3G%z^E z0|0m%&Ocq6rJ~j33gGw+3MnCvBTQo+R0zf33o)Sj5=!Dlw-pu;>dJK&RXeR)&WvI| z#u;r6CZDXzp!n!+3d$bY(20Z71qVH|7GN+wU`P?3IWXL$p?(1v&()5(fbi zaZD;Um=O6=!$8e#3X@e97!f846T6002j!IJf5;gEpbM<#?WxG%94ZKeit{08tS{s{ zvi%`bSs8dp{s2l_qXKXc$3)>g2Yd5EQe)05=@Y-wI?wNvY|0N(%-Et68a2#tsN0Jn zWx)_bjmTD1^Gi;ko|eVQsuZe>Hgl5z8)e&mnhq7C$;bd8^$_DqA79@6%&zPkDr=VO zcew^~H0w8UL9OgWw4?i?JfPos(6MCbiUx%CraB5@Dvc_RvNk6jbWT*Eh-7fYY$0*B zjriCnLI=$$x74(cP;wR&OY^8S9DYs(`12>D0tYX05nZ% zb{%PRmxi@=>yPG}RqKSOxl|~BBU%R{0a`R_$rtVfdR*E7H(zR*$RJmd*sFHVgj^A> znvv4l)@UaA0~A41Ha<2+&>zsjnHQ@lOnE*ZYX$mpVf9&gOPY(4$=wO}6;S9WMEJ_M zv&Gg1?>F!X*za@q`hWJ=IP=xkb1&}v=B^73yIzMlG0MJj!?tz9zI9cBm;d!gb}*a7 z@$CL?ENd28*hQd$!g6MYmO;eUS}0}w;HJVRI7jZvt*Za^p;&52toYKmMCn%$Cf@Mr zY;ROktLX5K12$XD#iQLV{fOter*-lLWaW zM22g75OGRGrC^;!5S9CL$K|)2e}29#wL0-kbM)RHN4Vj4b~=CmED>i6SWg#ScvC@E zx)hLE9A!BZJxF-;0f0YH$na1dVZQuB4#OX23#7$tz6-GU!vA}u8i2`{W8VGXPuHBs ztZ~RA;>N}w;Eyk_rG6J{hrz62nYF< zyz(|#jBa@ZJJ;|-VJBiuG_KxNc{YE+eq7I>&l@`n=>)Qopvwb<+1{CzUp#Ngp3l6a zSBFx5JfE*lOOz+&=bvi)@*HTQ7k~Z>ty+d6?J{MtXc2hoV_P)nfgiI(r8N^>70waHSS{LK9J1&XPwGvor@6?M3%*aTeH`6XphU+oF-BlM@&Yc17j`#L&aJ9+vQS-y7p=?OMiGm?n_nx-gufg zG$Xez9?OW}7(PJ|!74sz>8vMFRqq`~ruojqdRcppx})5fGi0Lm7Dft zddz~WLFzsOXCN;5CncM#9^VPLm#XD;(36*6rT&f?xd5XUo8X{iVkl1Kq!40v%cilj z`E%ofUQF^6tTetcY%<4SX0!>_(%@2pk}sbBliW8)E2ib?;Eh$&;ijzD9_quQegX41 zW8r-zF10)})8BPm7-t?|#LN~5=Og|PIVykXSpDU*OL1qH2)!*fY9n-HGu`H7f4Uh; zdhT?8z%X=!j_YiZNes8~Z4KdgWk>UEkK!gmen$zm3S%8Q5>U|0{&jQJ>`^3(EBdN z^^oxyPB4v{@ZaQNEW@>t(!U<-%b_dA_HZ8q!_R>3ZDyJ_C2ek0&NeM6DU3O!_mNSS zwiZLef;FP-6Me;-O)?Gwh+_)lwJS_pM6o_MQTa7oZU?Bq%Mc_kTqc~_^*jsJv*D*I z;h&75sa*E$mV~FA6Uprsz|R%qRQK^v+fonD$AB{%ofQTQpd(cpv@eLisj%yui=r9u ze;M%d;dSL3wK(TX{W7S%;MRn&ZBDYpLp_CTKR6i4^7@|yDxN;4V~y;)@5%@w(5-yG zNgDLkRW@LM*NhTp9kYaU7@M-7uVni%t9Cz<^*bM#R%H}~x4&wSDc(m06A=&yhZB$d7Q=)A zGH`)SyhUcf6@e$i%NGN{(&B!JKx3fYD~Ld4V?!3~3dPd0cD=Y2bp)i}4+NtSJ=Fq=pOPlJd*pX08e&jyar@m`o9bB13g-#f(s0 zl}QtsLHw;xOJSLiDcltVGj6@wpd^+=y@ncqGwm_)&bLVK!R$(O* z`HA)W6B-CLiKnjWM$eq~BzWeu30j&sNRiR5+CSXbI8oAO36XPT)+h?frr#v<9I``# z-X&X>(;Qgu9u(SiR+v`WyeAraVtDkfb$z02JM86JTV8#qYHWWRE!gVW>TPJ%crY;u z?AYpGaqK*1PWCrfHX6q^Wmsnf?$|mg{_?x}ujRPq*|N-o#nuvBgWU+qo5?f=0O-2h zt~bvm(gV3R(v={@`W$DgCEzIFf+_YWu2m=Oj`g;=q)}s!n9jd2Xs(p}02B0W?W3Eb zTU05ijBr)2x|H#$X%)M18APX!kkC`31$$F7U1)9jRH*ish)DN7V7qboYe*Ct-c?3_ z#t(p(z4FwWysVy!KbF+Rrzu=FM&2c-gcn8RMRe--A}yYc#Si~fq%YFL zvj5d$+TVKN`z95G(=~k?6-(FqKb3=)$T8Ejz7cnJwtAVu4>!OhH&ZU_VH(RMId}a` zAF%1+Wqq}zarEPe<$qu5=!bBFH3(c-T;LiY54=m{3VPgG&(HPN7Xw-YIs9u3#0=D8 zC0YwoztZNelSiJI=LHqXogJS$Y$NsRm@{MOjXKq+d7P@q zeORH-vc9>*$s(fx^)?O6TzXpJAfN*uqlIX3$qAb~!0WDWNJWrw859XsZCxumAx${}ON1M23?ug<%m@oVLJ=;qN*<_4>lU4@D z6ZhqNV2g%JA&3Qiw} z@g2Lqjk(%>bz>nf#Du9c@-a49ISR zUPA(q4d%4%!r~HsM~MbNjIlh$1!=aTs|G=?rSFE;PIRuhqj5fPgychOH{L(b%tZ`pXElW zrlkx%*D)uDVVkO!kgIx7S0>@}OM6yabMkqsyhIQc>0j*H+~oQ?W)v^0Qt$Q>(Av{u{Q4`&eGS zyz*d!_cItw<90tA7z2#LyJn%tc(H0q_&nELfLnSEx<}oy37)ZIx6n=_IYD1{nfleS zjM9tHx+HtB|9noftlBcZjWwI-$^NB4z@$-dFqMq;xNg9p*>AG4-M_SPchG9_=Jese zd0;6S5^2jFSvNTFo6nR?ZF=pzEVNd4cbnnwgZ|=aEr)MXB&UXvl&cobb;angy3ofG z9hRSDriY{-D@>AkN_l3{BoN{|90+KZL|Rc(BX(K`*s6x=7)D=KOH~qrXC{j(+qUUv z{oy|KBJwK16G64U1FfAAIuRc!fjCm)nK})1j4Cy8m5`KIg~NvPL0LMh_ClrW*SlpZ zFt*QXX414DCkg1*U>9*@($VwT6I2g!fR@Nonx4B6yK}ZeAf^g6Kt~r6EoGe;>%g`s z9w}k1TdR7rsWqr9qg%_eb@+~&v%%T|G>@@jw7wufvyuFEnxHrPuDNxspoR50QwH(g zR+m~9d@1LyRL`skRG?1%FScQk_z}(mZ~=Ys{$i*w5wWV(c7>aWop}Fs)eR0J=PWk* zbV-pZ5PPda3ufR2E8s)Jp^=CGoFbvPj!yc(ols8A`}>NY@oFI) z2#q!zMy((W_yFMXDNOJS$K9^Uko?`d`Be`95KNmSj0*K9KqJHTv-%av0@VSQGzpRi zc7yLj%${+Hmy@U=j@=@J$H)f=@X2ZtmgM56=ZjkM)7DJ$xyu#$w&UL%F#o$;!Yv8f zgN@M4_C#sA4?u_OxP>@j^bIqlnX~PU%v)Dw&muJmNu&KNZK|tnckZVG%fmf&+|W%q z#boX^V>vj}FnNvh$k7MNk%vMGAMr$Js-36$v~;KzkY#J{;6$+yRA7pyQ2L7K`Z6sL z6>)~w46{n*U8%w$fN*o@<8+e`x|^0BM)4i+7{#mM^fI@yx1xPa&71~}4O>Nkb(bt6y41u33NBjH3suzuH4daDKr(E~$LRE>LOGDX0vof4 zer~k*q@`u9I46A{vjfhjVfLn~Mak@uEAOSP=kikZsde1^2LN_L4gKf z4;vPPiK7pyRSqo%TUx?jL^7Bx|DVXE`a^$MXk1Z|Ay`*=+ZSO0s7(o(`Fi{zpEzi% zK5HcL#^h*od`=T0&TQ0;S`fsnqMRra&=IUE#S}~d32=B2kUZqRj6@&=1hK_yBs9n` z@F)?mEe*z!(JAsUc;?J6sFfnSeQ>7r38avBtV0K`Ll>|BRRef0u>9!Uv?n?~Ur`uiIRVKNyJV^DS7ha!{1zv*+zw zH*8B`sJ|Z<6hTOw2s6oL3d!qW2X803#b4jI1CL|1pv`IFHuy|Jq@t+1e=v}`uh*Kg zRT0rblbIgXapYj?hf(j&kp;5GO2FV|d~WuJFSW!ta5T)1l>S3EN`0&s7voKH9d@0U z>sw~X?9kpCH7kB)JO7Xq@rRW(S-Q3r0c4J4s!SRKFxQ0Cd7u0tA@fcL*!%%j-uf2A z3HrHtr88Uybm77%%LhomNVhq0+vL@>(QByk^M@1LSVOv*2c?`%Wm!`6lhqFJ$1_h% zwS7YQ=FMeODz+`3iqxVlN|KwOeDpn73^IqGoNLlHm;cnn^8Rd-SAff;r`?fSu<9px!@RKUrrH@=^2xM*74DRS2=_o|E$0`oB zFthSqsKRT^%F~?qt&I1V_f?iq3xxvtqz~mCvXVQ^)OL?OZKI9o4%lcy5G#oNv#X` zS=p9&XJ`GiPgP3scPm+IkRiCx3K9YEaBg&5-gX>wYH*aFye-t;SCm2RKc1}NXgW{bWf%< ztY%(BI#G%YIs;udL6vrsP4hpmCPuZ@t*EYMER!~d4yAu8X!A2m;^gvqEeYDbZ#8UI zTkgb~D_T5W@7emkzeg%gd$=ZNt?Vol)`akQOgb$YnfW|_`{?PuIPq|2UOSl+Otg*ZA8>1#D`MH@?}N-@3oCLP8#l~L$5r+%-qF}0hZL$k#F)a5y%nHChOECNtIrte``k>ECePc=hoBt|K zJV*AZY^mW|zi%ie4Frnp9GIWluvbu8?aaAAvwN64*HS#lEbD>KF%%D)O7JX1@7BG1 zz2}YBu~1iK^Y?L@F34Duu1I*`d|gmzJu`$gI|7ro=`XbiQHv* zsI>$_Fxg)k|xlp#NQY|X?P{&(acu*nShh&y7cE|w_B^^*fP@$ojlD2($|;$ zU$4UzKbrb{^-&%PBOOmiQlDSyz`QeH&t=|wV@FQx@+S-r{v4;r&qqbze=GlK*l(Gc z-Zozg`Fb7t(8&bi+(Ac^^+dAdZj^}3WTnvw2)uw%e86u~|69c0q!5aqXPil_c?Fgs z=sq<`1|H0LlSoUvHLT>sDt<>#>Wsu#d?kuRSGs&F&lmX4R-3xc_$@@k^TPF=cWA>@ zi@j47<{JNi5&=a1AdJvl(`(E=+hv;$hW(7oj3@G`Bv3|5GwX1U|oOQ z<^yFIbIhBovoM#zq&hz&gP)8Bk!VM|9EuO%&FyCsyYvDUDwEqH|kx z!uF6CXXTXdpo2z`ubkb!JX^5xmv1v`iQ=R$2w_3@O_qIw)FV}Xpc^k`xh~p!g?DS{J6`lKXeAR_&Mf*}Vd-9p0(dUb5{h-G1pX0dBLDI6_B4wQ8C{`yb` z{i#{ESouit`Z42D3DNrEVywPxW+j)B=chZuk6Y0_!m{jjoEVgE0W@onDZyKqDu3{; z)yNjiBxObBK(uKobiiLF{1YjX@L7UgGBK$%CWwrAMIzINSuP+b7JA+MG5X zX{6j_nG_V2d48<7Nd76|qmi|-;Jfkvb-6<@8xp!}v60@hVYW(; zx$zPe2X_!hDI8gGCp)TJQL#Cw=6`E)E5S})s=g(aieiLKpjz5}^$Zxj4ht_~J#_pl zZT#1>^B;0l!O)o+8fTyRs>7Lg4^@y6reuMlboW$WL*aRIa)#roH+>qR#<66{*2`@o zTZq<|zQ9w#a??ar>_CxS8^92~y{iuB;$S|PgtGHmJ_(sQx5!(5>lPn&lbIafMyHBc+^5tJ#(Pdb%mstN_G z!NB(#lNp7%Z*8JZAzCBxloRb-i^)E^Cr%zP7n(at0r^#dfy0Y@A4*?u*C;;@J&pSJ zQ-BH(@@Yrw)eCU%%Rh=@ZfWe^!=E`Ldn%xk)Tr_$$q^~0?I>jM5^+32I5a(Zn2(H3 zg7~6M)+2~oqkpETRrx~#5FnU$DpN85C5Pb(A}(mNa1y~Gfz-)9#OD7oXvEvnTDMwA z0yo3nZ=Q7TmlrwTdvqx;5j)yXfKH$iFT1v!A!fr^+FoS~tskB>6+d2^V(M2hrj%R(RK*6Gc zABH?O(8@}xwXN{*)Lpc*Ne!~=E1R@GTf~L!X~W!qBuk?T_yV0o9EQd4F%P)^JPAfc_|o!5F!R4)+H zOO4tmV7Y=Yp5hQrjdDY0Y))&Uhf3t^y(pBdA#p5=T ztIN7e42S>+NOTve8&H-Zln=u~Z)Z)Kox^)YaqzXfAs63{Wd7$2(2wKHBjhwHLHyQv zI6XCVgHL|Lb0Q?kw9-?pF4v|^=T{Z%!@$WuR_>@{u4qf~vC@o70NkUZK)CR7Q{Sel z>_Onj^w=2UrXQx$c(!Wm;-sw4sTg|8 z_qw0`{+FnO6^OeJ^Zc)Hp~jcH5z^|C#E z|8Curiv%f&L%u>OCr-Ug>*8UqA`lV|K*Q=^qB691> zKZCi}5BEnrzmbcQ#-)q-5p|v`zsK&@`i%NskgB@R^Hjya9-<6N^$QNIrTCiU@`yxn z8bz)msU|!cE<=lN9hA~tD1Uy=3~wFRnJ^(CiYeD>%Y-t8llF@mCQxje+Pssg zcfMp7Qf}vr@DxKeki7=`iyb|2Y-YXl`9R-aZ*uY$$?;WcCN75FZ+_kuE|+~p%*8__ ztjh}2U-s8_}o5>Hw^JL7TZr29gzwB9J#Gv^e?Hhg#zC^~<1 z0t1D;ZlL zn015tk&1Kj^h`MyC7GylpD?lag(07UIMOnaf_&d8i*kY>E1KxRJ_es4X_Mn-^m?DB zMO>R1fxI5yCMhtWTL+s|hqda5V}k7W^Rzthi(l{ByPxsr_%JND_wBaos_yd+xs=uV z^5;iV?Mj)ErOgj<`BYJmXKTwU^a;G)6_2UF)-7Al*(W=FtrgnLMb%oT3vo@UF*;eb z*tYmw>(T4`ZTaye{3z)(<+Iz9|Li_&RM}9tM27R#iFBI{N}o-5@fo^9N3)-V3ArhY zkzjt+Nz!KhjeAx5rI`k@$_(cp=iJ1_E3sueMvj2_Jqrc-AE4k5GlqX8Y+zpPYR6kc~6aVm1L`YRDTV^67SWYTJs>1t_3**Zg8dRfT!R?-3MV{Moj z+pB04uI4`VxspFXzUx_1@1;=c%jex3@a)iwlc{25Q499>AYl-;|Cs?KW|T3jsw%Gm zxmYE`jFSG^k1h&}WOtGHrR}T&-A>UQ(uHH<(H#)WP{R2U@IwHfX=wQG`XRmv&xkI1bEmckKNz>Q82%Y zM+^s26wPvE5oF>t2r_&_kz&8*I9C{YQ+0TbrwhGcz;B%*@Pj%oN)(&YOI7PrZAqj($vcS5H-|)mvNIOKUI5DRmlI zmy@?zkRytZC3Y_+3koCS$47w&A;ePs4Bxb?@^Hm51OMo`J+33VR)Ii?o6exou?QVy z5;7?)3l-1nocZ2eX2_6c*v=S1hMl#L7#W0pt3)ltz=JK2E3~K-3YuYpN_*ePTOPK{ z7%D#cm)xL`-x{3=d>8?J3|EH#-3A4d0T2Jq@)3KRI(oJ_UE6@(`YGf6|L5-icDPiW zHb{}+R5h=nusd%52YH0!fwi2lXfhO;2y7X$NMs;$`ABQ%bs`Q^eDQLIy9HDH`l1z$ zR=lrwa^MCb9Ml6TS1DGeMh<_F1{eT*c}NyJw8~6by|xk_8Kfkx!+gb8{x`4=2Es~& z%dgrR0h($i?WdIhCuWi9+OlI-&6U0$(1`Tv^=^BbfVN+djqNOYn1e6jQM*K`-|K0g1g3^-x(Z zEPhA~x;}Tu0QFG7rp(PITa(pn0S#u6sMuy0WT2NP*j-Y{JSA+PRZn55#tvmw%nuCV zKODL?dPiHcxXhG1rLBvyEz4)C>MnmXw%AL~Z0)-=&aA7c%gPmsDe6*mTF8#GOOA}U zEndob&UM^XEqO|oSAAXE1>zJ&cgs`cb50EfReqnf{A%-<%IG$p&8}Sgv$KMN44P;L z*?OlX3v}M!qiFpcSqW1!zV#qV=U@UiR{Bn-)$p?EhF-MhTc&l&;%aJUFX}p0yS8+& zW^c^ zt_+IvOvm06w;~vnbG(^!{iiHSvxtI7lq|YHi6M-HEfbrKhM}%sYCQ;%SR=!ui_WaL zl0+sKnx@WH-Qloq2+k~1Xlgx%?n09;Q@r~%{Pesa0$GyPaiX(6T&3+% zTyxz(Ps7AJ(RIGuwrB+wTRQWZ`NM>yO13gMEj4j32QhF>)$jcT^d$nbJ1-?E>{-*X zDw}r6WIE`7FYH7AA%YwP#F1K21bYv{cyy8+Z`$}@hY>o&)gIgiaUG#}8f;&Fi(u@I z+`@YAmY$2J|8Z?=y+$&&D9Eak*uvf0n#_<+wOf2>@z-tbwm2{eCG^93_Mgf503caZ zS3m?!IKN6Bn(`WeR;Q$RXG9=4UV#-Q(=DK%(6^E8CN=z#S^%im?~M+nWBZYfiusA6pXK`jJl;*^0z^jsG{!%&mCrdv9E}J%^G7{51 zA2DUNEU2`WK zd%vey#KQ5T9cijy-Ka#z#xuc<>d2TE#X=9s)+38=)EVd-IHCIBlkv`sxGn5-ES-sK zM#Yr%D{e3-P(O@je}rH>BtOeLWz-*chx*%OZDH!M<9+X_!A>t@ui3bIL~31|_V+i* z4#w~q)|pri+4uPaYD`Edk%6<1PhkI30vnwG0;O!(VSDE!5|3?9&iSZ8D>&;ooKiw$ z22&soU}U;fEwnHaB8VlD6^XkLOYQ;@A5f4<7V+gPQDkU>wk126Hk$?vGTD@(Spxj; z0^@~3sy2kOZ~hG_UHb!p5=5p<$N&uDB(PL4Y&utTBb1j-Eejrl5IS^KwUgiBT3yFd z>^zJU);h1jBLy%%@M%)Ey?0my%h5JO*AjqoaA5}4UdVx(QHVf>QHQk%qR#3SmaNia z5`&weS%_0_H`dS@bC*+M-wvhov$!Rht~H0ziXwz?!7%TctwxVsf>*`z|7hsmfiQ0t z%h#4ffT*#ouxU8}iaDoYPZ|KMuaveusR1CEuoFIe)Z0pAwTe*`KY|Y!!osh@f@^zX z|4m~^QJT(5n^+g{Ax;2X)*A`X7WinN;@$C-<+_xgmlHXbJbWx*Efgxohs4(vujxRp zs8C}e+TS_!gd~EX1U#NoI4FjA{t!dPMFvO4BWkkJmV_t6zr%MB5>Hd4QNPm!bPI}c%+ExEbQ6t@p{AIZvQ;@D1R^woS3U*_}Wo$;8cVdAP>Z+_m(6DdC`+; zD-k0AxXTqV5f#GCsN!2>j+Vy=83hD*G%-Cscu_-+EG{H6$^0T}&|;?4pIVg1sk%ke zzmz%=H(fhCcVrG&59?gI2=pH|I~V%)Z_3f`p7$E!H~nkp&;4Gf0@mEm_8;c!Ik`X5 zE>+t~GpkWFdix^u=0q*6b@o8KMObYy@vR~v zwP`j8QlqcnE<+)rD%xasO=4hV(1m})5h9Oau6Q;aHal)Q_UvtoPnKF+Tnfi@1@oF` zb}&ZtRvZo;cAip-X%vx6+c)VQb$yS3c{ZIfS;bFV5iMt$`<0o~4}$c8AUg)gI2d|D z-(-HvmkkdIO95f|PZgM`l#L&b(3W|JSh&&<~(uN z3Db>nqsmDjCPc&WJq5kwZ0)u_zNU1^?-Z5%nWC-ly8nT`Yb(n1sP!Di|DTsrgoHe| zTs{LutQJ(S-pogOYR1{xEl&VgXrx+!SS3)r0??gmdQki-It+QRzm1Tm0HTs*snHP@fKoGy*-jShJlyt(aUM%03jAk{mnX6yUws*N@r z&@*o-ZQZp~%WUw{5VO}c&ciR z!7pZXyIFkDJ)!A+I;sNhyr%)bYhBhpSvIA%8eVB5oD&5Ca@TZ+;Qjf>lfEK;BeK$F z__+Q9h7l1T)kH}>Cnh1AV-qnA1xwVCYV^8F4p)e>yiu(j1;y(5QqHauQzZrxy~0F< z*1f!cd%FBpX0J9FQX-e$vlK%o$9{*Rk0Tib0gw!Vn}8eKtx*0s9rcIx1RmFxSW-a6Ou)U5Du5#Nh zh5uSB1Phe2(reXMne;1jTA&Dn^U#0aRvz~LvMClm%El@0T2w;_Ca~}#dL~Sg(8x_- zv1>9M%e|&CFyym=Lv1nDOGJVp#;2lW_BjHINLr_0{(EiCtpIXEcARLF15A!pX zo_RvoF8Wi0Rx{@<)!N-dG}N%xe=lYkc;fB*LN|CqY?1Y8`alOT2uY>x}eqy6J^;~1wrOWFGjlITR)8B);?Ga+LsyrJ`oorshxIx9Dm0*aS!z<*)lH!9*3^Kc^h{<{!cY9aGo> z@E8hnl|2+BWu|5Rwe*cH%$u!!<|LKzwIAft7x_>A;spS&)niFq6W8m+mCU&UO!Y6~ILNIG&PY10#;z};g6NneV^*uqD^7EJAc#Oa)5`lZudPMI!z)Ih0Vy zAEvsF%tYY!R5A9)pv(vMe8ciy#EE9&nijK6r!Rq~j!&;CiPh<-bFokPNmC-BY0Nn) zJ6em<^klYPW~V8WF*Xz3e%4wPms)QY4qR23KJw(PBou}Dz}565GL;T#^d`;pOllo0 zlM1~AEtR>{Wkw)6O|x_}nM$g?^h5of)}k-&h62rqlD4u58>kN;Vo-FyuoK$IQ&JxAakSf3q1rQPOArx z#yzeoWI1Y6W<}?`w)Xk^3-3Z}({|Fc13#%F3aqo}BFln(tUX!X#Vt6Pv@B;gs}6dk z9gk5t{xTEvA~&)%KE+Q~yjf0*AeO)q1Em4`Ffr;!qJh|#E_P!ZEyRT_Y)A6KHP0wU ze^hd^Jh}*Ss!CGSF|s;*Oc=<*-+eSy(%sA=$jeH0F4{3<57XQbM%Ss8&#Jz8iv4M+ zoosz2uZ=`GL7PWJ&%#6vuc?|GydKu7(8(JV^w%>iDJ#%TTPTzh1pr7^f^I)rSZ#n> zFccSZmC@M+AzTHGUtX2j7aeu7QuqB|_5GZJH}@>wa13@qrH9j1E4J~Z`~9#}QVAv; z8i$)t*`}s1S$P+GrC9P0@Ut-c56S>r$HSTTv$bK z&tp}lkj8ycsI4q{ykL@HwnbTf`->c@Xn+YN0&jdXPf%&&CRGF{L%l0?&*g(>clre$ zU+L(R;A3O;%u!gMv6YOt$HRi>M_Frp;M0r6L-EgX%5qX)lu`Vg;o=JSdUONPXarLO^rNMXMq5HPpet z*E7e*rMDNjr7Jf>Tw7MKIaISX>J#U#y7aDU;WtGSpj%cl!}n?%z{8=-j!tLo`-bT& zvQB4}af`}@LvxNEeW`Lk>A##ts|9KzitVE^wC56avI9eE2Q@5HIvn9qU5JWt1~6Kv znobI`iXYVTp_k^mbQT`y)UA(1DtCsd;v8-H3u@aBelM%HisR zv;;rUz$^-z*ZHIwr?PRp01Q?f+%8d(>Vz%Ym!9yt*@r6UjZx~#qO~CH!NC1o&@=CE zdPd;d8pfB5!})O`NJx`*rs$!;LY&}>9-R?z8j482VzF9ulnD^nMBJg%&Mmn&{R8Ol z_|e06^DsTdz6<5=S(XKbo^g+ThH{&#AebM}>|+UZ4X`_C;A1RvNB0zar{B_fTtMmo zZ$v@oi=LGj{RjX_Yk^)K0ASbw)1FjkR9>5$Aj)OWd*iJ;8llv*Fy&P1a<~wdR}u^6 zkkQv{3Tg`?ic^m`yV&Rsi(4rlI`SEFd3|o}#&55wY?#7tIR9F*|I#xku;XPz_Ir_5 zhjJobZ_h^4x_xUsGA?r-58s(2S)PK54Q=FI>eLK-lh?7g0f#cZ+_SxE`5dnzw}v%1(rZ|0Iy=qr6lN|*6XO>7qawtB~Kvix;r&B>Vm%B-c^?b}|xI;wM?@RD0b zJKd+W0NInahSLxfmlA_FmACo4Wj-~tH=ALi3V^yd!{lBOd{&9~Pqn_S1zLJuy+$De zAiMiTOX%Fl0!g4XLWs|1M!JR|3vs5#h+@d%S;w=O#3UOVo2d0Uy}Dbd{>~QbVt0>w zP0RO*)DiVO>B>?z{eH-!Nt8wto1s+4l?bxBlYnkcl%GwBZ}FbPbg^zFize^&dP&s( z5L{nle>>O7k~7TCvr6JPvTAQO{Fnr`o36I|)BhWb0kumSbtU3Gx){qz`!V7tf{Kun zffChJ;rS8Dt^PzBLku3j%p0j@&kc4iX_vI$u(|dzAV!o&P3DD1!f6wV5%|=@Bd*nR z9Vj9vYnsj$T~tqgO`Zq+u8!LLrN%5S-ViNRnovWQ)_RY<64m;e^O}$q`0uf>E}OG+ z?J|Fec2P}<#GvjDNd{=KP=7kZyIAuW#_VSjfZO0e;*@BG0X_RT{#2Imqw5&}cRcJkZfq3+pwU zuzc~=xA58XZf(T<`35dT8N#HdSD9Rmi_InuE{rs?r~Sj;>0hN1RU9Wym*k^jsbH}1 zz~bSU#IT`-e`2Advx(5@aXHO*Xk?aY0luLAM>@d-Q`WB>3(80#zB%FsYFncOpFg7Fc2+T=YwZbG2Kn+M~{{= z4ikSwj*~{@+?t4vnqlj@K`*C|m@1iy1Q4XZlEOx1QdNYO(6M2WGLdM&y$Lkmdz&G0 z@(hr=SVv_(NV79FDme?MCq#8u`$1bL1NXQe}Ms)6ooR&G4%sY|vB&|#|NitpKksAzbN27}=^_6XbIR!c zyVUs05m>5QwlxTi?8rK`L-p%cWGA9?DanSv|5hQ<7+o^xuK(r`hF6h34=M8gWqmJN zvkAM`-Ktr({=CDsfOt2Wk5p^Zzl_lC45=y_`8yG2xsNP$>d;b~OJk+}SZo#cqL4XQ zNtEQKuS-`-hfuxUwFhT1Im4N2&@>H4Y-!HTji>j^k$wcx_nU5gvPTK-dO{IBFk2)+Q{Ki|-DJ#e?U!^F0?@{OlF zh-?f&ch9_~UK0UI06?^a-o_~Tw@}gxhDL7QYG_GeB_=RA=}OpjvI>in%o>ij0ff8B zx7gp^pomIWg3x)L_h3}O8oavIR4!UXWGFSDk z@_chTZy)I(d077VCBDn)g$yj8-Lj3-yaLaQi7bW|R=E1#MdoZGHu4_@nKWYZ)LU#D zw9t9faNvvbMjaJ!k=MkTtSGn1?A*LCl_sjz=I*`queMpXhnAb0WY$)6E^Bl%0UPs^ z?1Xle%5hn|K}L2A+;qm5Olt0%)%2P){eKiZ>UYV9(KS9x_WOXM^{?QC2=95EW1d8qTny_YNS)`hf;}!J)&DJP(o){bs0v<+o z+*Ilq_PJw9;S&0=B;Z`=z2w)5XS=7fSKAE+{lCtY~99%ywm8sgjJw(FsAG<@hZ z<#H-!6`>hjqt)sv)Fc|%B_)?)VtVPcJd41UB?DdfbrZsO*VnsPQ+%LH|9wi;hrN#7 zs8kAt?}nC}4x6b!LQf(9syLa1ULoNnSYrE0-0 z0lYS3z%j{Y;2rp9DiI&iI@^ys9;=3Y*p^(CPe5cKBY*}s$?b2FaK%d z@NX0;mDyHDtiD^(!4gNvk7afr!uQdSSwoigXqI^jR|HN$3i-cLFAgLSjQ zEq58I6c3TRb%)da^po#gaJ?u4-6@U|lD6sBy%sZ8v84Ea!| zj??hdm$te@*Q8}<{}S8-g6M3FFQUOW(=`6xiokUsuuKLdl?`THhT_m|?huG}lq%&~ zD1pkQlPEFVIJ-bZ8Ms3q`l5n(taSz*+>3A-vJ-|1S=EwNn1~EyOJ0@!4a3m0r#-Qww#z2YR$+Yky;|d_tyGwyE`EbgSmGMyBfE*=iGV z@lW%iWQ4}@$a~F3OQoW(f1(Q+K5(z0njiB0OrAj`$UgQ~RWYlL5T733x=4T|sP3TX zdA+_)N{l@`@h?~)7ztmzpj-p8s1k4Zy#NojjYVj0ypLwP+65~@+#M@-LtuyK{_)&V zJ=HC+RAhV{>ebsLH+D(9a4q|wEo)--JiUYJ($havqwCYY%YH6&in81fw-PpD^y%lzL zZ^r-zPllVU(*xkgJNL`!-WzATFHzB=WMPOiMB&S_MyggiH5Bo&aE@p2b&1@5G4wh< zQ6CwyE21b?Aj(fnqur2k#EN|RCHQS}@P!6gB~#T1ZAirBrXg(1SsWZZUX9=K$jqFK z8Q7UP1gtiFY8FCoME<8hsC=X44iOUP!qjEFF3E`)6{+7WBGuGa6= ziIJ_3P0rFVz{4>DUKi-HyIkYbm-y~K0Q_3@=5pBfr;pUf=iO3wrTqc(@x~R^*HR~} zv{5Z%4Q^}}QlO0R^H6c3ng~>N$oBRRJ@X>+T4R)-+NcWPG8N2UDMF;li(l@S#v_un z!sp*Pp%;I1bVR+CH-B8a_IF%dU!(BybmO0jt>}8Svo~)WvaIef+DX@(h~*>MIy4Vw zAre}pxnI%I1r-$(mU7?{Ue${}o_TjfGf6>qI($xdec$jSDV`SKU?ue?hzQX*2A|RxK*}2DrgJcFrX6nmL!HC=XHQvVV zwf`>sMy(+Sd5oT=##ny@mUoM0i$fkey`TLY`zUAR?$+P6K6?M_#)InxMrh=;F>ivC z`<$Xo16g z+Q2jUh&AncKI6h^%Wyn3#|eGW)~>R{n{xRonrwYO)7Plu8PK(-tFk#`p6y?5v$Am& zY*Mt_zh^Q1r$w3s#i&Yrn>_{9@X41Oy{dzV84LiPZr!?h)wX5bsf)|+6O<_$Ej}dm zH`5)JpFgV20yBM^kngMCI8~2Ch*MhxiX7xxPXs7`zwsiV1)P~%H=(rd2HUl5crQbg zmqObrJhboP1UbWiNIYcyPU@60jOuLpar2o>atWC z&x~8ZeTpzRMKDY2Z~G{<#HO4w*5x93JFXTgpeg_Yq&P3JpE;W|Z@eXik#v-X_i&ClW9Qrd`8{56nEyKo+%G2MEYHGNyr^5)6T!varZ$BTF#lm zfzoIP=V(MG;el?AaOhC`&Eq1tILI-Gq%2g^bcfY+Uf)pY7)o}?(~-V$84#GeOzzE0 z%l{y9%c1&FGJa5RIM-eFm@qIyg*o0dg9|Knx6z4}M=y3$^SzPFjsX)(yf_A#eafQp z%|*?MK?`(6S|0qN41#o-{p~_E#Gk&**3j40@Vnk)pNpQ^E4tmrYK})TYh#I+fGp!D z`YU?%nm(;&ON}^f`b+vwHopFlG#e!whQ%cW>(T{dG7lt}c8Q zSG`=hMdEYKt1@+-=wcTdzff)ysQT`*R;(m(*aAOMHwQkcl_bJHJ3!R}D^hjc&4;I~ zHZm)s<@zQJ_69pS{qVP3m3Z{NmBece!1IJPLp2gb4{m#&=LteuXvKPl>}amNY~;8sFGPXdYWJDP91 z8~1tt2&4AVU|u*yvimg*+P#U(d!nJw@&+DG_g_gjOpEdeA zGb;Or_iq#*Oj7vj$ixa|$yR=ZVckEIqk)8CHd#5lk#*51d9`BxEtA5ZZ+OW!GZ9`u z&z6ki6sa^`TLO7bqsUzsTPUdCx+M$~jX}A=6mM|)E6pa+83Ng~U^@7!65ZF$8QzmN zMLFGis+hQhQ;N>Qpf9bzQ=(gqY*Nrx+|(OEN+T)MUcl8k!N1NgH%+SewbI zIDT8wAH=3YIAA@4``?R@!)R)>_`pPohUMpngZI8MO%faD?y7Ih5kkdu$3y_;htbgk zxj3+pu~WDjA#R0Ei#K}>L_n)RmxhHfNy<7N&`4LwYiB zK_FGg7?M>}wZhWPXe{VAxcq6yXJH)NrLl;vDA76pH!SM*9Ha=aTn2HY&$KH&TwN&nwylS$W15Idp}i3D`+oiKst|&1JoLKXi}a#vEq@^M231c>B$|6fDmEsy0hPj#vjD!Nx_6q;^@1O*1W2g=5nq{;#SBSvxJ#Mkt*QQ53N!KQj;g+zZkOY!7}9~eclz(U!BQ7mLO#0r5}e%@ z(16=H!zS`rz#n#a6~efp68*LKo@(T5?lZpJxrBM*>nFZy8>^wO$}>~z2#vSO3cvN0 zl0G4GF~J7X2A%zZr{7}N&(i^mVtUWMzy0dHDqt}p2{`w6&+)V0kBP%qH}*iZVe!AG z!l~TgHnk%clVyI@M&*I7aE3>b(id)^x5gQ@Y5g|8nop~3m=kI3VZyb3)8Q3{3Ce0{ znS83ZjWs=vno8Q+XlyOLHph`5SR9qG*_*|$9`|;z5h~$K(?c9Z9W8ewZk0{AOstQ*Y zWa!Nxk1JQO%y&(CZb_Lv>^8mU$hH{!8Tt1EYDxwjHv8_yuS_ey`Y*V94itQ zRklcpC?`!sgNuwjFcB4CA7AGIpSK~NSB)N(^|M=hsch@R;SJv9U`CT5JLd}TywisL zhdi1nSNEG>XS%n`_|raY5A&lPgizp~<#89{{M>cY2 z1nJ-HU*tqO@AnSyhwRYdg6aj#HMOI&wz5 zKuav|ol}x2r}9%r{99V8h(NI2?pf0#u))%U^wjEyY>GQ;Li+%6%U_x z041?z$iV4?B33kRMeUyFK4CXMNQrSXeVkf=i>1*TPFhvi#UwR+$PCn zlf=^qx)fB2$&wy}@B{M@wu#EnzDm`olPu;l;3NA~ft|jbgj4Z9I zZ&_xFilXXre$nK0KcCaCp)bu62W8dyD(lNeWoDa@3u&5CeWGar5z4u});hj78pIFL zYZ*N$@^2yBBy*a@GOBUu1TIv9+L@hw*Wo-`%hWb%8*ExW!}OD1SJHl6#Ti$+%h}&A zsp7XMvCz2$>R<~2C^li*xptCBxxTDXB8sEY5~78L!lMR~5KEO(UKjxHh0(%gti zMae&*M$~X@X~yABWO0jC6mSOI8K(f zCvkqSLf_P~A@&nNm9MzQqUXe?Jm5#!i6XVLIjqsId&pKIAprV<{f-ztn+ptL6+FkZ z^uGYCim)Zodtftl=S^|ssh^>_p)Mz_DB8-XWU*TOG`B{UAnChAp8Gxf2oxUFpMtV2Ld1o1N*I7{VQRyD%H#5iFi?L9ge9N; z@a}{+SF6v);(-b8&Xm?LBfe)V-waIdCzBp?(Mg}yOf2fF(U)dtm(%=qXEarT6}Nwc z7yP;wg^TfVL5l9#n)Iz$U>mEUk>A=57<0&HFB(NqQyr5F-hK38XJ_Z82(%{d`7m?d!q>KlB|NjX@OyqmaoSb=Xp~ zjOI|6IaAUmOO3RqM+g6l8JqAc;Xv_bb%LM}nbQ^vYm}u65k8g^VB@G@V9t(rmWUOjVRkX{_t^iqjlUx-%NT#J4W#C(sTUBl!$l z|M;de1+P~d&%3KTrhm%b|BLrpcXU>sAOAO1Ijt<6<95I-r(&JK|1*YwlxP31vOaS0 zqGAvd2xbrjK3|X+aV_w1;7-(Tkx|$70C=5TVgT!rvyqHKU_a}GzxK%P@Dt| zIvX+smx2*h%?C}U*3JYcr+yG0b5YP}fTEvX& z?d$NV>vD3_T8TbF9u=+qO?e{n#QPN3A(~i2GkIH_s%py6r~B~7mj8o*P|EN2_UZ7w;=v`(5fWWQ%JO1Uce$e^}#tABD z?ttG86#S7@61#tyJHouI)-&$RXQfg3^>`{)M6Z-`*y3X3_xDCtnU}aRmC7l^xn?W@tQtPcePaD5~r5P0V*CzTONQBfhO-_BWoGY)v zv+4Jq3)35Hd(hc>KML@9djUbgAAP-M8OD>#P}O4K0C38>jbde~CM0B3DWb%Juu{t? zc$z|bQA$+N`9%;PZ#tCIc*$bxdxtMu_6VVEm0&z$XSu4rF9m~6)1zeB`iX)*cdPFP z5beFur@gaLo*%-23N=g*4UuXikhT*q;57;t8P`Nlq6mcRnSai2n9!!!xI`f-P$IA| zm6xd;UveRznkzRVY^x$3N{w1)CPtVBqn$W~pn4k|`u)mGqhRh9@!3gl3w~xv?kLO>ds^q>&st)KLw3I97)UcnY!>I_}h?Q*@=)R=#$ow7R1h9Z$f1K)Fw zYfF{CEnoj_U~1-uzCQo*S1VBGW8pfMs*eQ>l;P(&cT=TxCK_C%avXSfzY1|09dy1d zG=8)aCH6NC6V@#8&v94l`A8>g^F~YFbyl^8_odk~wmJrQAFW#!cCiifqcLzNOal_RH$yl`aHD zNGEb`ulWTrz3ZRo-kvQpw$pUKJ$ZFchPMjv4C%`sDJJq_&CAA{=rR?*N@OWwBAlM6 z__N}HeH*@+?Yi1P`?=Kq=lXsJ!t3Xde z-_@PpEA!xR^ixkS2^0xd3LYXN0#S<-6X+@Ak(FQ3S)^qkG|*xv#S-pd9wTyPeBYMt zvs!fshUGqXffg3{LyHHr_#y&>V$=Nrk!^*F@3Fru90`G+6}LyBP(xNqLJ7u&n<6{F zVmrzCtZ7h`2(F@X{Sg)Hom8EMh>{K>n zG|jWRrLV}MiXXzx<=qgBklkywm^K5Ff3}Th70b_#>P{5fes3&Ga6M;71F^V(Of6*5 zv@-RaIx43$NiAp6L|sNKR9W}ME;D{nY#rH~9c8?rZ6NE8%ehX7^od4KW57U63{=%j z+ZG3NiXgDZRxs^l7+?KNDSQ{|k1b5*!awCfax!X2>7JK1-PYOwKtFGQe!fnwr(c25M8DJVXw4q)@)3VP+tf z+PWnf(W{Zzh)-KwiLS)?%pHb{4#W1RQt25Ry%9)j45}T!%W9F12OAwB4~L0~+)Ayl zjP!sB05~v4F-kB38#TM%;F~x$oU$4@>zttJ*Is_Mp{l!N?h9y?jdw#ZW08?dn>LCr zmqxunqsY4cJ#&IQgzVwBotTtap0f6XBmaMOnec4mmz!43$^Oy~EnxZ{B61?(# zQQgCZb-j0XZL)-#0TYa?gb6EUZdE$bU;3#Z(@oa_z?%TUQ(%Z!WWL31NDL9AIMNro z94@?m6r|sQ02nD`(a>|l=8Fc<{2j`??JDi*%i1maH`Id8BuMEDC%oUTqy#aAMG7(F z2o^_S2w*>bi$fOx5zXi6nsZ=ve%0izKO(;xYuJ9yfw$CIYZGXnP)^HEWKt83Shnd@ zy&{pV7>Twt=E?gTGUb^}j{CGM95#->;wS8lMIE=SW@Fcu&$>ot$3zR%hai49pIP#vQjaD5rFQ6UQX(@4juhNodZ2t*?Q2J zQUZ@G`E~{~o<56^N!pTiS>7_1*)sWX@*n|Ph znXzzTh|o!~k!CznTn=1IylbyK1wgGrDxh5( z2!6t1`%evBTWoIt92Z(BLOeGO7oRU>#ylf96 ?6gmLtC>C49V`2-y?Ck-|; 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

-$`?Cp}t!=_O- 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 T-T%@5{O`m6AHBW*!x#TA1IT)D literal 0 HcmV?d00001 diff --git a/app/audio/buzzer/baby-cry.mp3 b/app/audio/buzzer/baby-cry.mp3 new file mode 100644 index 0000000000000000000000000000000000000000..669dd1220b97ad91647c664c2313cef86385a0f0 GIT binary patch literal 122912 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 + + + + Wire + #include('../../page/template/#dest/style.htm') + + + + + + #include('microbar.htm') +

Colors

+ #include('colors.htm') +

Buttons

+ #include('buttons.htm') +

Media Controls

+ #include('media-controls.htm') +

Checkbox

+ #include('checkbox.htm') +

Icons

+ #include('icons.htm') +

Popover

+ #include('popovers.htm') +

Avatars

+ #include('avatars.htm') +

Dots

+ #include('dots.htm') +

Grid

+ #include('grid.htm') +

Modal

+ #include('modal.htm') + + #include('../../page/template/#dest/vendor.htm') + #include('../../page/template/#dest/component.htm') + + + + diff --git a/app/demo/template/avatars.htm b/app/demo/template/avatars.htm new file mode 100644 index 00000000000..0132a2e84e2 --- /dev/null +++ b/app/demo/template/avatars.htm @@ -0,0 +1,289 @@ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
lgmdsmxs
Initials + +
+
+
JD
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+ +
+
+
J
+
+
+
+
+
Image + +
+
+
JD
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
Selected + +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
Pending + +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
Blocked + +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
Unknown + +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ +
+
+
JD
+
+
+
+
+
+
+ diff --git a/app/demo/template/buttons.htm b/app/demo/template/buttons.htm new file mode 100644 index 00000000000..6c952c53155 --- /dev/null +++ b/app/demo/template/buttons.htm @@ -0,0 +1,100 @@ +

Normal

+
Button
+
Inverted
+
Disabled
+
White
+
Outline
+

Small

+
Button
+
Inverted
+
Disabled
+
White
+
Outline
+

Icon

+
+ Import +
+

+
+
+ Import OSX Contacts +
+

Round

+

button-round-xl

+
    +
  • +
    +
  • +
  • +
    +
  • +
  • +
    +
  • +
+

button-round-text

+
    +
  • +
    gif
    +
  • +
  • +
    gif
    +
  • +
  • +
    gif
    +
  • +
+

button-round-dark

+
    +
  • +
    +
  • +
  • +
    +
  • +
  • +
    +
  • +
+

button-round-theme

+
    +
  • +
    +
  • +
  • +
    +
  • +
  • +
    +
  • +
+

button-round-theme-green

+
    +
  • +
    +
  • +
  • +
    +
  • +
  • +
    +
  • +
+

button-round-theme-light

+
    +
  • +
    +
  • +
  • +
    +
  • +
  • +
    +
  • +
+

Icon

+
    +
  • +
    Import
    +
  • +
diff --git a/app/demo/template/checkbox.htm b/app/demo/template/checkbox.htm new file mode 100644 index 00000000000..598911df6b2 --- /dev/null +++ b/app/demo/template/checkbox.htm @@ -0,0 +1,4 @@ +
+ + +
\ No newline at end of file diff --git a/app/demo/template/colors.htm b/app/demo/template/colors.htm new file mode 100644 index 00000000000..ee8d98b3885 --- /dev/null +++ b/app/demo/template/colors.htm @@ -0,0 +1,11 @@ +
    +
  • blue
  • +
  • green
  • +
  • yellow
  • +
  • red
  • +
  • orange
  • +
  • pink
  • +
  • purple
  • +
  • graphite-dark
  • +
  • graphite
  • +
diff --git a/app/demo/template/dots.htm b/app/demo/template/dots.htm new file mode 100644 index 00000000000..a9390db1e21 --- /dev/null +++ b/app/demo/template/dots.htm @@ -0,0 +1,7 @@ +
    +
  • +
    + +
    +
  • +
diff --git a/app/demo/template/grid.htm b/app/demo/template/grid.htm new file mode 100644 index 00000000000..34c35c88c34 --- /dev/null +++ b/app/demo/template/grid.htm @@ -0,0 +1,5 @@ +
+
col-12
+
col-6
+
col-6
+
diff --git a/app/demo/template/icons.htm b/app/demo/template/icons.htm new file mode 100644 index 00000000000..6715c7f188a --- /dev/null +++ b/app/demo/template/icons.htm @@ -0,0 +1 @@ +
    diff --git a/app/demo/template/media-controls.htm b/app/demo/template/media-controls.htm new file mode 100644 index 00000000000..b14d3b520a2 --- /dev/null +++ b/app/demo/template/media-controls.htm @@ -0,0 +1,38 @@ +
      +
    • +
      +
    • +
    • +
      +
    • +
    • +
      +
    • +
    • +
      +
      + + + +
      +
    • +
    +
      +
    • +
      +
    • +
    • +
      +
    • +
    • +
      +
    • +
    • +
      +
      + + + +
      +
    • +
    diff --git a/app/demo/template/microbar.htm b/app/demo/template/microbar.htm new file mode 100644 index 00000000000..b6c725db037 --- /dev/null +++ b/app/demo/template/microbar.htm @@ -0,0 +1,9 @@ +
      +
    • blue
    • +
    • green
    • +
    • yellow
    • +
    • red
    • +
    • orange
    • +
    • pink
    • +
    • purple
    • +
    diff --git a/app/demo/template/modal.htm b/app/demo/template/modal.htm new file mode 100644 index 00000000000..0d37e524a5c --- /dev/null +++ b/app/demo/template/modal.htm @@ -0,0 +1,15 @@ + diff --git a/app/demo/template/popovers.htm b/app/demo/template/popovers.htm new file mode 100644 index 00000000000..76cca4932fc --- /dev/null +++ b/app/demo/template/popovers.htm @@ -0,0 +1,20 @@ +
      +
    • 2
    • +
    • 3
    • +
    • 4
    • +
    • 5
    • +
    • 6
    • +
    • 7
    • +
    • 8
    • +
    + +
    +
    +
      +
    • Item
    • +
    • Item
    • +
    • Item
    • +
    • Item
    • +
    +
    +
    diff --git a/app/font/Wire.ttf b/app/font/Wire.ttf new file mode 100755 index 0000000000000000000000000000000000000000..447b84563523bd0aaf87019ce1e9d05a4128709f GIT binary patch literal 8996 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 HfAIIe><49s literal 0 HcmV?d00001 diff --git a/app/image/debug/ping-fireworks.png b/app/image/debug/ping-fireworks.png new file mode 100644 index 0000000000000000000000000000000000000000..1db5a695f017796eb3e2be6b6b2e855bdc7fb81b GIT binary patch literal 33010 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 A{{R30 literal 0 HcmV?d00001 diff --git a/app/image/debug/wooden-background.jpg b/app/image/debug/wooden-background.jpg new file mode 100644 index 0000000000000000000000000000000000000000..6abac9e97002096fb86c54e8c6ffb546200a4ebf GIT binary patch literal 293648 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

    }ca`8SqV9EsO1f#9=K=n&6=Z_<~9 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

    wwmy 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 literal 0 HcmV?d00001 diff --git a/app/image/icon/wire-icon-128.png b/app/image/icon/wire-icon-128.png new file mode 100755 index 0000000000000000000000000000000000000000..3282867af4d83eb2881d504ddb64b2d10eb93351 GIT binary patch literal 3986 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 EKL$j#KL7v# literal 0 HcmV?d00001 diff --git a/app/image/icon/wire-icon-64.png b/app/image/icon/wire-icon-64.png new file mode 100755 index 0000000000000000000000000000000000000000..801028a663991b379e521122b44b4a81eb8f9e38 GIT binary patch literal 1832 zcmV+@2iN$CP)Nkl9aBoc?M?f4z*YqfsD529pu%t)d zR)L?V4hFdJ59toebVjf&-GzSu2h42a`L{hR5}nQg(Xsfp*UDk)-i6XJk*Ns~xpWM| zG~YvGe84SUBAJ>5lEuq_d(sse<6pjM39L{P!HSl6`8L@Ktk0_^u|Z7+8*0+$1$MlD zTlu=*y*VkAufYb)=W#Vq|gO`l;n1Q4J^27DY_t}B)EX>@gLhpGF=czj`d_v zFU}f(BuSg-8&9u$qIa-7Yp}L$o^9K<&Hw(hZQHhegSB^iSY7JQdr@ChRL53VjA!~y zW^BI9FTc0tsxKX(=x36Uj`}5{pCN?wq~rgNo#IANb+y!{G6`WD$_#Q|1Q>f_H*mez|v5Oh>o6-$a@#U^FvN5RlQ$ z79djh8@jnWA!BL3xY>LGCCscLAf+u35I2|hLk86K|EmMchL#D45-k&8Hhgt}x-X<5 z^w-UCyQ2lf?Z0kD2o#Wx9;@FJ>1Y8WtKUQq&W;R(Ysx!AKutNqoSlCWV@qcUFt#9~ z1ms&Aa05A67GUXq*r(yWZdu`j6 zAoP?TDfz(y`BJ_R@L$QiB{ZU&^b$K5_Lujn3lT5HkDqKbSmHvp@EYwVkX0%8G31|- zr|8F|7`Kvqu^0PbG(3K?x)i4&R5mnjT4D-Kd76h~1aemr@7C=_{!lrXes_;|!4cp{ zOM?2w&9b2gsk>no`fvP~=>WXvoQP2$e--&W0&=M@S1sF;f7dTefy~B#(cgVd<6-#T z-KK7e;Uo>mlvm}G1aiEO|3lIQ4u15FJB9276)Sk-RITye-OeHCK9Vh-X`6-w;V0?F z$sX|W#|w!4ef;B+E?p_h2!w4ep6NZ3r6C*JtoX@F7@{Y^a7^JP$q~q5vdraaPslC; zVq1AQiCyI!GuiT-VwB&iua#Kw6ShIN_f(vLycog)O3VUyrJAs1&%4797pA9=af{);*{Sy z`M+c@$rYF1k%#1FRtLk74dp0KdNEP2X6Sw6*`^+o#_+GaFE7fcoJ3=!DNx*J+)eLN zOz=pz-qde63%$2~J>|FcX> zt4nbn00#n(c zd-IoUyh~FKA}Ge}3$Cd6%`#&=ETdBGnDS9yTql}tnH9fXa0O;_WP}}f*=g0X9l9W& zt!9ip>cz*`R@#=PX?BCNhg|_Rj2c9&wrl;uB|^x<4H5K5 zF-Bu5=3)tU$KJ&5Si)#3*+>QOjdIjum1z8 WuT+Z-zt%4R0000|=k~)%rlZ-3 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 literal 0 HcmV?d00001 diff --git a/app/image/logo/wire-logo-120.png b/app/image/logo/wire-logo-120.png new file mode 100644 index 0000000000000000000000000000000000000000..504b73e725be607076bc0f47f9a6c9f1d97f9989 GIT binary patch literal 6467 zcmXY$dpy(q|NnFOMCDYe6rxL)=*I{rcnmIy_#_=i_;QXSlgKs_oF;v1Q8^ zwTlM?%TWaKu9Gpj60yKZ@C)j7Y*L zlC@tW57cyQiuVP)WqlKt?1gp-2<4I41FI3!{?f|L0GY9Pq2JR!7kTxNm3EiXZ1aH8 zHk&BtZW$SmJ$>t`=}Fw#BQb}YLvN*rM-UGEj6`k#wm6;PVV;@tzj0eFWn;T+>9%#xA0oQjrpWv~V8@G;0PFdgE9e$AfPhb_@5j?M0E33is_LJcU-{ zRc)XudQlrdZq$>I!q`ITw_V7CIGmfTUlG^Kv}CNnale=lJPDL25?c98xYLIf+{x!# zTPbPC>5ttz@Uqjt7>y((M>B9z7tcIos%Gr~(zk7;e65f1`#Y5(!~X+}XhIV**N)`d zJhF@n%qIJhi>K+!%luF>7zYGQHP;T1hd|ssux!lOMh*PA8^t-rVuwXh5fVxoPrw;m z6&%3ju(H{~ddBuZ$b%5D{x^}&_n9uyOFZcGGr~_%HO7PUsrd3m7uVUQH-S0%!dO>O z4%mn=s|}QeENKH%gE|xvQjG2bqsDJ1T}SV9RezpgeLZTmpSAPgQ9Q5Wu<|FqFDw)x zWT0cl_R#pt5NJ^hBajOow1y`yS}u(_Kg%&;^$C`RDqECU{i0#% zJK}jn)L%DKJ1wJNS4u@@x^8$v1Il)^m-f@~S zu)0DJa4Ce-qiF8nM$=r74IG;ySRR0FKcf{g1y`JHe##DEtT@MJ^ZGx;XIs~76Tns8 ze0`#e7Dh#e&u~Usz?h7x=t7&ds$GLghicRdOSa!jH;YlUv{ROgM07fMY(9Ycq zXtXTEMx{tjOrVPDwGIh9Ha%OMAm&ftF3D?oY)K7N=&{*UN{pPIp5_q|sJ8H9f>rIN zE55sTw&1c#A-Lk*J=^kp4QYJ4*VelHeCx1XPY-|EhZ5i8&%QEQQP6*XZ5(WUR5DiS zwA&sqoOFMvVifym0D^x8kJSLm$`>f-10ZyFYz~+#3bWlXhPw-J`(f8 z&l5^&-dgN2yD?tW-s_yGcF3fqr3`*k6q<8|3y8lj)EP5iTqWSH_Xv}6sC?~LFUP7) zMG2L7iIo(%7WHzn&1&c3zXK{G8a|GTS>Wd&2Tq-G_t!=OayXjpJ`GKZC zKMpQKGz@*waT!4ebu~d zpUjoMWMaygwiJ3Ul=lZ^)){4^bsF}aVwn)?%v2(tXtLI0$MRf&xNaiPQ;nW zy!3+J%XiySA2H#n6vavo=l{05XoAsm$swH1e-%^Y{*UpIrKh8e(@Vqnw6%4|(pi0R z_r%`aGxnJoJyGiJ{=7tCdSVO&c?-t^2C)ktqq;peUcogcy@8M6STB8jAS}! z7{w1qI|>&fv5gztAlXowKM(y+Hdfq6ScHmua@OWZh3I!1(;zHZWE;^l3OpTEqR|U+ zU%U@q$f<%!=HOXg?fTdxVH#}nuMa_~x=G9Qk!&mWkAG^uk5O>(K5VQ^GNMIWZ(EeV zO>nH*dlWD0bp|a^#Vqm~v2GNTD6AzXSBBP9iB|fo{CJ+f$Fh(53;Bs`Ug2WnsAO?s zCj1n{kPtWEJK~@fXtDmrM!p)2NG2q{?B!pv>?BEd5Q*exhd?nonvS3ros z4e~Q{*ujvkHw4V-uNz}F$wgrug#i(TE$XLlUdAj@gtx6mlXzD_t23`6Z|4QXbQpY+ zcg3Qen3lCc1D8gaa{n->o1(v7r^I7wuWXXg7K|R0OH=LgJJ#T;w1gw1$s{G&Tz%Bk zlf@;hY{Q_gT`%!?LUsbg9Q;l_8)>1Z8EMiKE~sqUREBAI-Gw?I?Na!6<>hnK0z_Ha zJm_7R(%=9mBc=al-3T^EAm)wnRWJ`KWf;s z%l6R^$^swv*R9RvDHj$np~_Jh}S2 z#PLnr9^}2b_9owlh-i6G5nb;4Mxt@vUNPm?U?Xz0@-dm3E#)-&j%u8yqM4GJPI>V@ z-M*PSCbEJ;#36_WoV}~EeGI2_{5v3+pyuT)jR!MG)@IkE{;)d5tk6I88H^kMae8J; z;j!qZ<_bd0*i(>MlHJ53zNhNJS(|mAM=Bcbitlb$=z}>PIU4Ee&oRr}3M0~^)xfqs zeXNQ8yS*u#)!3>|ij)-^Esd2NJ-yq6_r^rhgiK;=x46}9U{^IU>t?X#ihw?O- z4K#hNkH{VOU`~eJhSnOHH&XZTnvfsufvT4+aSvn|m>H?M#3XRf)fZ3g_M5zY!@HZH zGW`6yNn7~q*IS85M7d6uh8{iSiwyY0ICetpx{m;{2VQaCLQ{BUyactRihbGg&wNME zBq;$Qkh<}D>K_K)8wF#RaCuF8kZMh(CT;$_=i#V8lg)f@=m zpko1$xS|=H01^uaIrYuQ$ROhAHgB|pjfb>cYuQ9*xWWX7bT zQ{QN?hBf@zz?S}dzH--Qz0GENcKpLsLQz{9nvdKGfA8tfbyST&wS#1PUX)fTLBn8*{NvX+v;t)YB#?KnL2H^GOw@&wt?ng1|CA>o-pKgf2Dqb0zpIIO4M< z4m+Ts^ZAKA(8)e$sw+sSHUh=EQXl*sftp}1Ro1Z$0v!1E0OCx*2jfD-D;%;5Lfhb5 zCA_2iS@35-6u!F*`#fWzfm_wBSML*|)vuBdr;WDulH2a4Xjg+Ct?U>8a?2QVQXNsorU+Qn{#EAB4Y*!(?O6iA3`r{xA{cH zb}be*E_~8iRp5N3aRZn9S_=~w>)ARQ!t)l>n^IXh0VAHHJ{NtYYr!33W;JV|f-ci* z19T4yyZ@cs$vi*b!QxjoDa{z9FEuVv?@>wNF9PfcntfrVZ~NRou}&f$kZp26VlsD9G_( zoM%qg<%T=0`6`<^4jMGcLZsF&c+!OC2qJNzcKsYtT0@Y`+rQRXY1M@D!-!VYqE<`~ zK7{`vC%NL+`mchU%3=QhbKFMBzw4XI=;+KWcv=~*1n=3;a=UI(l5rT!VR<%v@=W~o zPn}7xcw|Tevz(Hgo!E2MK6)RU@8P;B&sq+`Z^y`oxYXrT&~fXjA0auZx8p~Xg(XG^ z!&U-nch2d>G;lp`#e^6-B*_{GsL8r{ zfk`WEka!Isxq6vyi5{XXQ8)JeDVd>A)%voR$u#ePXb0g?1X;~=ZI~y|e@pVjmw%lm z&;K?O@D7vf23E>tdu_<5&q~w<*V7BJ7b_1<+I$`uR(TAq9~8DUtZzQ5!f>UjT8LSU z+|m$Z2c#gZ$T@zs!$`=xWg8aXZKoVZ!o@g|Xlf-yZw7!vGA5pthazlfg$3jW&bvhq zg1EmVfN3w@(0!G7gY9dT08t+J-XA%`n`r@HGLaN+vMFL>5e*BIIWRUwNLgA?2Twg_ zpB?=Yh1}bnz3jcxX(49BQ~Q{g7UBj22W5K|)?VpskY>L_?u)rsNYaGuWDM%sEb~}J zb*$oMUh_&XA!Zpic_!&kj&p`yz0}bQBjE(Frc~A0Cz~B`Of~Hy`cFcQ1u#gH3Z!t$e`u1m6n6@7z5A?299o-P zIkZXG7zP^t^ARJ(X?hJuljTd&C|{e-kxme;DV@cM7>umK^t`o4)3PIfJC3L$$E7%p zhUZJgdV<8!4Frk&>NsUvR*)()vflfhtHXlhtwMAwE+iO?)=8DV5Rl zISsjrIBcrs)9c~g!hd#adGvbVo$YdT0ew87>UhaOTNnlPUaF0t18bw&3^`a~QZ8WZ za7=+Oh{Bvosub6gc(toREJt&JmCqZCqIO*0j}S&jUANr5a{?xtN`)jaFgwyB@|#{M zD;CC5z0ml2Z05{zxPhy=HOkU-#l6Dfn%Ma)`6)Dnuka%NKeFZwBDX1;n7`okWh zoIGi7cljIUeu~{RoU>?@=U-6~XKW+VweqE{^!Yq$Wm8i3HIqcN z=r}QdHDP;?oE2qjtlvv;D^0+rM5z6JzR>>t3rFv>YIo8sqL#cTT1)%GBE#nZ>p?i8 z{~EzKF+8=)?xFY!urS&YQ&W5M&fpJ)S;$t>x@hagB`l`}}Zp$DJo=j}I?>EQP4K8cPuUl%>=zDrKM}X@38@?d1ZFE zIm`K=Nn6ub)Wy?=GhtDP^0;hX0dx{t!|P|QA>?}+*%tSJUU#wbo)$QvW71L%9IB!bedPGz zuqfz2e2t9qx6(nKr#Nh2jH@1~Sgy!PTvAB{?ln)3(P7l^6~zfpB+$WV;e8EumX}-V zAIAUdAc8k4;1H-~m$;u8#4#tMuWLri&$4#VT#i+~pe&xen_gEfBySEMQHwhSMwhZ9X2-4Nf}h}==E6oo<)Q0T z(M4ldb@;;g7k8@0P@0@Gcn$U8U;-p>j!dx)kGLX}V-mR?SNx>ZMX&j5hIhsUE4p!) zC`POdAa z0AIG@$#Db|7E|tEa#+$<`aXHm5vDX|VZ zCJz`*Gjh-zL4@nWvE_doi8~UgiysVXvID&R6_ICF|lOSaq>lU?8F-e4JR bi|^Gl&L2a<%jADWwp?^@Jy&HPeDD7NxI^<| literal 0 HcmV?d00001 diff --git a/app/page/auth.html b/app/page/auth.html new file mode 100644 index 00000000000..233813bf43c --- /dev/null +++ b/app/page/auth.html @@ -0,0 +1,360 @@ + + + + + + #include('meta.htm') + #include('graph.htm') + Wire + + + + +

    +
    + +
    +
    +
    +
    +
    +
    +
    +
    + +
    +
    + +
    +
    + +
    +
    +
    + +
    +
    + +
    +
    +
    +
    + +
    +
    + +
    +
      + +
    • + +
    +
    +
    +
    +
    +
    + + +
    +
    +
    +
    +
    +
    + + +
    +
    +
    + + +
    +
    +
    +
      + +
    • + +
    +
    + +
    +
    + +
    +
    + +
    +
    + +
    +
    + +
    +
    +
      + +
    • + +
    +
    + +
    +
    + +
    +
    +
    +
    +
    + + + + + + +
    +
    +
      + +
    • + +
    + +
    + +
    +
    +
    +
    + +
    +
    + +
    +
      + +
    • + +
    + +
    +
    + +
    +
    + +
    + +
    +
    +
    + +
    + +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    + +
    +
    +
    +
    +
    + +
    +
    +
    +
    +
    + +
    +
    +
    +
    +
    + +
    +
    +
    +
    +
    + +
    +
    + + +
    +
    +
    + + + + #include('#dest/vendor.htm') + #include('#dest/component.htm') + #include('#dest/app.htm') + #include('#dest/auth.htm') + + diff --git a/app/page/index.html b/app/page/index.html new file mode 100644 index 00000000000..9b8353d1ca8 --- /dev/null +++ b/app/page/index.html @@ -0,0 +1,26 @@ + + + + + + #include('meta.htm') + #include('graph.htm') + Wire for Web – modern, private communications in your browsers. + #include('#dest/style.htm') + + + + #include('svg.htm') + #include('loading.htm') + #include('wire-main.htm') + #include('#dest/debug.htm') + + #include('#dest/vendor.htm') + #include('#dest/component.htm') + #include('#dest/app.htm') + + #include('partials/template-user-profile.htm') + #include('partials/template-confirm.htm') + #include('partials/template-message.htm') + + diff --git a/app/page/template/_deploy/app.htm b/app/page/template/_deploy/app.htm new file mode 100644 index 00000000000..337f3b493b8 --- /dev/null +++ b/app/page/template/_deploy/app.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_deploy/auth.htm b/app/page/template/_deploy/auth.htm new file mode 100644 index 00000000000..3fb89d9d2a9 --- /dev/null +++ b/app/page/template/_deploy/auth.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_deploy/component.htm b/app/page/template/_deploy/component.htm new file mode 100644 index 00000000000..6b3b183fb34 --- /dev/null +++ b/app/page/template/_deploy/component.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_deploy/debug.htm b/app/page/template/_deploy/debug.htm new file mode 100644 index 00000000000..137f2aaf167 --- /dev/null +++ b/app/page/template/_deploy/debug.htm @@ -0,0 +1,55 @@ + +
    + +

    Conversation

    + +

    + ID:
    + Participants:
    +

    + +

    + +

    Calling

    + +

    Tools

    + + + +

    Traces

    + + + +

    Soundbar

    + + + + + + +

    Debug Info

    + +

    + Version: {{config.CURRENT_VERSION_ID}} +

    +
    + + diff --git a/app/page/template/_deploy/style.htm b/app/page/template/_deploy/style.htm new file mode 100644 index 00000000000..15583123479 --- /dev/null +++ b/app/page/template/_deploy/style.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_deploy/vendor.htm b/app/page/template/_deploy/vendor.htm new file mode 100644 index 00000000000..31a868eafd8 --- /dev/null +++ b/app/page/template/_deploy/vendor.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_dist/app.htm b/app/page/template/_dist/app.htm new file mode 100644 index 00000000000..7f97bf19c5b --- /dev/null +++ b/app/page/template/_dist/app.htm @@ -0,0 +1,277 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/page/template/_dist/auth.htm b/app/page/template/_dist/auth.htm new file mode 100644 index 00000000000..8a5eaaab884 --- /dev/null +++ b/app/page/template/_dist/auth.htm @@ -0,0 +1,4 @@ + + + + diff --git a/app/page/template/_dist/component.htm b/app/page/template/_dist/component.htm new file mode 100644 index 00000000000..61e6d89e40e --- /dev/null +++ b/app/page/template/_dist/component.htm @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/app/page/template/_dist/debug.htm b/app/page/template/_dist/debug.htm new file mode 100644 index 00000000000..8eab3f16808 --- /dev/null +++ b/app/page/template/_dist/debug.htm @@ -0,0 +1,61 @@ + +
    + +

    Conversation

    + +

    + ID:
    + Participants:
    +

    + +

    + +

    Self user

    + +

    + ID:
    +

    + +

    Calling

    + +

    Tools

    + + + +

    Traces

    + + + +

    Soundbar

    + + + + + + +

    Debug Info

    + +

    + Version: {{config.CURRENT_VERSION_ID}} +

    +
    + diff --git a/app/page/template/_dist/style.htm b/app/page/template/_dist/style.htm new file mode 100644 index 00000000000..1cac98ee618 --- /dev/null +++ b/app/page/template/_dist/style.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_dist/vendor.htm b/app/page/template/_dist/vendor.htm new file mode 100644 index 00000000000..519f37d6c8a --- /dev/null +++ b/app/page/template/_dist/vendor.htm @@ -0,0 +1,44 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/page/template/_prod/app.htm b/app/page/template/_prod/app.htm new file mode 100644 index 00000000000..337f3b493b8 --- /dev/null +++ b/app/page/template/_prod/app.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_prod/auth.htm b/app/page/template/_prod/auth.htm new file mode 100644 index 00000000000..3fb89d9d2a9 --- /dev/null +++ b/app/page/template/_prod/auth.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_prod/component.htm b/app/page/template/_prod/component.htm new file mode 100644 index 00000000000..6b3b183fb34 --- /dev/null +++ b/app/page/template/_prod/component.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_prod/debug.htm b/app/page/template/_prod/debug.htm new file mode 100644 index 00000000000..e69de29bb2d diff --git a/app/page/template/_prod/style.htm b/app/page/template/_prod/style.htm new file mode 100644 index 00000000000..15583123479 --- /dev/null +++ b/app/page/template/_prod/style.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/_prod/vendor.htm b/app/page/template/_prod/vendor.htm new file mode 100644 index 00000000000..31a868eafd8 --- /dev/null +++ b/app/page/template/_prod/vendor.htm @@ -0,0 +1 @@ + diff --git a/app/page/template/conversation/connect-requests.htm b/app/page/template/conversation/connect-requests.htm new file mode 100644 index 00000000000..c50061f33b8 --- /dev/null +++ b/app/page/template/conversation/connect-requests.htm @@ -0,0 +1,18 @@ +
    +
    +
    +
    +
    +
    +
    +
    + +
    +
    +
    +
    +
    +
    +
    +
    +
    diff --git a/app/page/template/conversation/conversation-input.htm b/app/page/template/conversation/conversation-input.htm new file mode 100644 index 00000000000..bcf1974934c --- /dev/null +++ b/app/page/template/conversation/conversation-input.htm @@ -0,0 +1,62 @@ + +
    + + +
    + + + +
    + + +
    + +
    + +
    + + + + + + + + + + + + + + +
    + + +
    + diff --git a/app/page/template/conversation/conversation-titlebar.htm b/app/page/template/conversation/conversation-titlebar.htm new file mode 100644 index 00000000000..ae98c26e33d --- /dev/null +++ b/app/page/template/conversation/conversation-titlebar.htm @@ -0,0 +1,28 @@ +
    + +
    + + +
    + + +
    + + + + + + +
    + + +
    diff --git a/app/page/template/conversation/conversation-verified.htm b/app/page/template/conversation/conversation-verified.htm new file mode 100644 index 00000000000..2286caff533 --- /dev/null +++ b/app/page/template/conversation/conversation-verified.htm @@ -0,0 +1,8 @@ + +
    + + + + +
    + diff --git a/app/page/template/conversation/conversation.htm b/app/page/template/conversation/conversation.htm new file mode 100644 index 00000000000..87975fbf1d5 --- /dev/null +++ b/app/page/template/conversation/conversation.htm @@ -0,0 +1,22 @@ +
    + + + + +
    +
    +
    + + + + #include('conversation/message-list.htm') + #include('conversation/conversation-titlebar.htm') + #include('conversation/conversation-input.htm') + #include('conversation/participants.htm') + #include('conversation/detail-view.htm') + #include('conversation/giphy.htm') + + + + +
    diff --git a/app/page/template/conversation/detail-view.htm b/app/page/template/conversation/detail-view.htm new file mode 100644 index 00000000000..ddf8d0ef580 --- /dev/null +++ b/app/page/template/conversation/detail-view.htm @@ -0,0 +1,6 @@ + diff --git a/app/page/template/conversation/giphy.htm b/app/page/template/conversation/giphy.htm new file mode 100644 index 00000000000..ea65612d2ee --- /dev/null +++ b/app/page/template/conversation/giphy.htm @@ -0,0 +1,46 @@ + diff --git a/app/page/template/conversation/message-list.htm b/app/page/template/conversation/message-list.htm new file mode 100644 index 00000000000..64071785094 --- /dev/null +++ b/app/page/template/conversation/message-list.htm @@ -0,0 +1,18 @@ +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    diff --git a/app/page/template/conversation/participants.htm b/app/page/template/conversation/participants.htm new file mode 100644 index 00000000000..2c87d9c5219 --- /dev/null +++ b/app/page/template/conversation/participants.htm @@ -0,0 +1,74 @@ +
    +
    + + +
    + +
    +
    +
    +
    + +
    + +
    +
    +
    + + + +
    + + + + +
    +
    + + + +
    + +
    + + + +
    + + + + + +
    +
    diff --git a/app/page/template/graph.htm b/app/page/template/graph.htm new file mode 100644 index 00000000000..9491a50d4f1 --- /dev/null +++ b/app/page/template/graph.htm @@ -0,0 +1,5 @@ + + + + + diff --git a/app/page/template/list/actions.htm b/app/page/template/list/actions.htm new file mode 100644 index 00000000000..8f9ffa2e908 --- /dev/null +++ b/app/page/template/list/actions.htm @@ -0,0 +1,60 @@ +
    +
      + + + +
    • + + +
    • + + + +
    • + + +
    • + + +
    • + + + +
    • + + + +
    • + + + +
    • + + + +
    +
    diff --git a/app/page/template/list/archive.htm b/app/page/template/list/archive.htm new file mode 100644 index 00000000000..55128bcb56b --- /dev/null +++ b/app/page/template/list/archive.htm @@ -0,0 +1,57 @@ +
    +
    + + +
    +
    +
      + +
    • +
      + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      +
      +
      + + +
      +
    • + +
    +
    +
    diff --git a/app/page/template/list/conversation-list.htm b/app/page/template/list/conversation-list.htm new file mode 100644 index 00000000000..7c1ebfb303b --- /dev/null +++ b/app/page/template/list/conversation-list.htm @@ -0,0 +1,199 @@ + +
    + + +
    + +
    + + + + + + + + + + + + + + + + + +
    +
    + + +
    + + + +
    + + +
    +
    + + + + + + + + + + + + + + + +
    + +
    + + +
    +
    +
    + +
    + + +
    + + +
    + + +
    + +
    + + +
    +
    +
    + + +
    + + +
    + + +
    +
    +
    + +
    +
    + + +
    +
      + + +
    • +
      +
      +
      +
      +
      +
    • + + + +
    • +
      +
      + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      +
      +
      + + +
      +
    • + +
    +
    + +
    + diff --git a/app/page/template/list/start-ui.htm b/app/page/template/list/start-ui.htm new file mode 100644 index 00000000000..3dea8f05a5a --- /dev/null +++ b/app/page/template/list/start-ui.htm @@ -0,0 +1,167 @@ +
    +
    + + +
    + +
    + + +
    + + +
    + + +
    + + +
    +
    + +
    +
    + + + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + + +
    + + + +
    + + + +
    +

    + +
    +
    + +
    + +
    + + + + +
    +
    +
    + + + + + +
    + +
    + + +
    +
    + +
    + +
    + + +
    + +
    + + +
    +
    + + + + +
    + + + + +
    + +
    +
    + +
    diff --git a/app/page/template/loading.htm b/app/page/template/loading.htm new file mode 100644 index 00000000000..4965084d584 --- /dev/null +++ b/app/page/template/loading.htm @@ -0,0 +1,6 @@ +
    +
    +
    +
    +
    +
    diff --git a/app/page/template/meta.htm b/app/page/template/meta.htm new file mode 100644 index 00000000000..48075416912 --- /dev/null +++ b/app/page/template/meta.htm @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/app/page/template/modals.htm b/app/page/template/modals.htm new file mode 100644 index 00000000000..55604b44a19 --- /dev/null +++ b/app/page/template/modals.htm @@ -0,0 +1,287 @@ +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    diff --git a/app/page/template/partials/template-confirm.htm b/app/page/template/partials/template-confirm.htm new file mode 100644 index 00000000000..e8184a3fc2d --- /dev/null +++ b/app/page/template/partials/template-confirm.htm @@ -0,0 +1,96 @@ + + + + + + + + + + + + + + diff --git a/app/page/template/partials/template-message.htm b/app/page/template/partials/template-message.htm new file mode 100644 index 00000000000..c99ec707ae1 --- /dev/null +++ b/app/page/template/partials/template-message.htm @@ -0,0 +1,235 @@ + + + + + + + + + + + + + + + + + + + diff --git a/app/page/template/partials/template-user-profile.htm b/app/page/template/partials/template-user-profile.htm new file mode 100644 index 00000000000..eb1aee716a3 --- /dev/null +++ b/app/page/template/partials/template-user-profile.htm @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/page/template/self/self-about.htm b/app/page/template/self/self-about.htm new file mode 100644 index 00000000000..135afd1d659 --- /dev/null +++ b/app/page/template/self/self-about.htm @@ -0,0 +1,21 @@ + diff --git a/app/page/template/self/self-profile.htm b/app/page/template/self/self-profile.htm new file mode 100644 index 00000000000..a4239d15e13 --- /dev/null +++ b/app/page/template/self/self-profile.htm @@ -0,0 +1,28 @@ +
    + +
    + +
    +
    + +
    + + +
    + #include('self/settings-menu.htm') + #include('self/self-about.htm') + +
    diff --git a/app/page/template/self/settings-menu.htm b/app/page/template/self/settings-menu.htm new file mode 100644 index 00000000000..c8fd06f083f --- /dev/null +++ b/app/page/template/self/settings-menu.htm @@ -0,0 +1,8 @@ +
    +
      +
    • +
    • +
    • +
    • +
    +
    diff --git a/app/page/template/settings.htm b/app/page/template/settings.htm new file mode 100644 index 00000000000..dc5767a2f1e --- /dev/null +++ b/app/page/template/settings.htm @@ -0,0 +1,173 @@ + diff --git a/app/page/template/svg.htm b/app/page/template/svg.htm new file mode 100644 index 00000000000..d31f5e4ebbf --- /dev/null +++ b/app/page/template/svg.htm @@ -0,0 +1,84 @@ + + + + spinner + + + + ping-color + + + + + + people-color + + + + + + camera-color + + + + + + call-color + + + + + + video-color + + + + + + giphy + + + + + + + + + + + not-verified + + + + verified + + + + + gmail + + + + + + + + list-call + + + + + + list-call-unjoined + + + + + + missed-call + + + + + + diff --git a/app/page/template/video-calling.htm b/app/page/template/video-calling.htm new file mode 100644 index 00000000000..7a2f0f4608d --- /dev/null +++ b/app/page/template/video-calling.htm @@ -0,0 +1,99 @@ +
    + + +
    + + + + + +
    + + + +
    + + + +
    + + +
    +
    + + + +
    +
    +
    +
    + + + + + + + +
    +
    +
    + +
    + +
    +
    +
    + +
    + + + +
    + + + + + + + + + + + + +
    + + +
    diff --git a/app/page/template/warning.htm b/app/page/template/warning.htm new file mode 100644 index 00000000000..5daa27a9cc5 --- /dev/null +++ b/app/page/template/warning.htm @@ -0,0 +1,153 @@ +
    + + +
    + +
    +
    + +
    + + + +
    +
    +   + +
    + +
    + + + +
    +
    + +
    + + + +
    +
    +   + +
    + +
    + + + +
    +
    + +
    + + + +
    +
    +   + +
    + +
    + + + +
    +
    +   + +
    + +
    + + + +
    +
    +   + +
    + +
    + + + +
    +
    + +
    + + + +
    + +
     
    + + + +
    +   + +
    + + +
    + + + +
    + + + +
    + +
    +   + +
    + + + +
    +   + +
    + + +
    + + + +
    + + + +
    +
    + + + + +
    +
    + + + +
    + + + +
    +
    +
    + + +
    + + +
    diff --git a/app/page/template/wire-main.htm b/app/page/template/wire-main.htm new file mode 100644 index 00000000000..850f535f87b --- /dev/null +++ b/app/page/template/wire-main.htm @@ -0,0 +1,59 @@ +
    + +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    + #include('list/archive.htm') + #include('list/start-ui.htm') + #include('list/conversation-list.htm') + #include('list/actions.htm') +
    + +
    +
    +
    + +
    + + + +
    + + + + + +
    +
    +
    + #include('video-calling.htm') + #include('settings.htm') + #include('warning.htm') + #include('modals.htm') + +
    diff --git a/app/script/announce/AnnounceRepository.coffee b/app/script/announce/AnnounceRepository.coffee new file mode 100644 index 00000000000..13d9d9ef396 --- /dev/null +++ b/app/script/announce/AnnounceRepository.coffee @@ -0,0 +1,75 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.announce ?= {} + +CHECK_TIMEOUT = 5 * 60 * 1000 +CHECK_INTERVAL = 3 * 60 * 60 * 1000 + +class z.announce.AnnounceRepository + PRIMARY_KEY_CURRENT_announce: 'local_identity' + constructor: (@announce_service) -> + @logger = new z.util.Logger 'z.announce.AnnounceRepository', z.config.LOGGER.OPTIONS + return @ + + init: -> + window.setTimeout => + @fetch() + @schedule_check() + , CHECK_TIMEOUT + + fetch: => + @announce_service.fetch @process_announce_list + + schedule_check: => + window.setInterval @fetch, CHECK_INTERVAL + + process_announce_list: (announce_list) => + if announce_list + for announce in announce_list + if not z.util.Environment.frontend.is_localhost() + continue if announce.version_max and z.util.Environment.version(false) > announce.version_max + continue if announce.version_min and z.util.Environment.version(false) < announce.version_min + key = "#{z.storage.StorageKey.ANNOUNCE.ANNOUNCE_KEY}@#{announce.key}" + if not z.storage.get_value key + z.storage.set_value key, 'read' + return if window.Notification.permission is z.util.BrowserPermissionType.DENIED + + if not (z.localization.Localizer.locale is 'en') + announce.title = announce["title_#{z.localization.Localizer.locale}"] or announce.title + announce.message = announce["message_#{z.localization.Localizer.locale}"] or announce.message + + notification = new window.Notification announce.title, + body: announce.message + icon: if z.util.Environment.electron and z.util.Environment.os.mac then '' else window.notification_icon or '/image/logo/notification.png' + sticky: true + requireInteraction: true + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ANNOUNCE.SENT, campaign: announce.campaign + @logger.log @logger.levels.INFO, "Announcement Shown '#{announce.title}'" + + notification.onclick = (event) => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ANNOUNCE.CLICKED, campaign: announce.campaign + @logger.log @logger.levels.INFO, "Announcement Clicked '#{announce.title}'" + if announce.link + window.open announce.link, '_blank' + if announce.refresh + window.location.reload true + notification.close() + break diff --git a/app/script/announce/AnnounceService.coffee b/app/script/announce/AnnounceService.coffee new file mode 100644 index 00000000000..7f8ca1dbb70 --- /dev/null +++ b/app/script/announce/AnnounceService.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.announce ?= {} + +class z.announce.AnnounceService + constructor: -> + @logger = new z.util.Logger 'z.announce.AnnounceService', z.config.LOGGER.OPTIONS + @url = "#{z.config.ANNOUNCE_URL}?order=created&active=true" + @url += '&production=true' if z.util.Environment.frontend.is_production() + return @ + + fetch: (callback) -> + $.get @url + .done (data, textStatus, jqXHR) -> + callback? data['result'] + .fail (jqXHR, textStatus, errorThrown) => + @logger.log @logger.levels.ERROR, 'Failed to fetch announcements', errorThrown + callback?() diff --git a/app/script/assets/Asset.coffee b/app/script/assets/Asset.coffee new file mode 100644 index 00000000000..1d80fe30e15 --- /dev/null +++ b/app/script/assets/Asset.coffee @@ -0,0 +1,64 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# Asset entity for the asset service. +class z.assets.Asset + ### + Construct a new asset for the asset service. + + @param config [Object] Asset configuration + ### + constructor: (config) -> + @correlation_id = config.correlation_id or z.util.create_random_uuid() + @content_type = config.content_type + @array_buffer = config.array_buffer + @payload = + conv_id: config.conversation_id + correlation_id: @correlation_id + public: config.public or false + tag: config.tag or 'medium' + inline: config.inline or false + nonce: @correlation_id + md5: config.md5 + width: config.width + height: config.height + original_width: config.original_width + original_height: config.original_height + native_push: config.native_push or false + + # Create the content disposition header for the asset. + get_content_disposition: -> + payload = ['zasset'] + for key, value of @payload + payload.push "#{key}=#{value}" + return payload.join ';' + + ### + Sets the image payload of the asset. + + @param image [Object] Image object to be set on the asset entity + ### + set_image: (image) -> + @content_type = z.util.get_content_type_from_data_url image.src + @array_buffer = z.util.base64_to_array image.src + @payload.width = image.width + @payload.height = image.height + @payload.md5 = z.util.encode_base64_md5_array_buffer_view @array_buffer diff --git a/app/script/assets/AssetCrypto.coffee b/app/script/assets/AssetCrypto.coffee new file mode 100644 index 00000000000..3952aa39be9 --- /dev/null +++ b/app/script/assets/AssetCrypto.coffee @@ -0,0 +1,84 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +z.assets.AssetCrypto = + ### + @param plaintext [ArrayBuffer] + + @return key_bytes [ArrayBuffer] AES key used for encryption + @return computed_sha256 [ArrayBuffer] SHA-256 checksum of the ciphertext + @return ciphertext [ArrayBuffer] Encrypted plaintext + ### + encrypt_aes_asset: (plaintext) -> + key = null + iv_ciphertext = null + computed_sha256 = null + iv = new Uint8Array 16 + + window.crypto.getRandomValues iv + + return new Promise (resolve, reject) -> + window.crypto.subtle.generateKey {name: 'AES-CBC', length: 256}, true, ['encrypt'] + .then (ckey) -> + key = ckey + + window.crypto.subtle.encrypt {name: 'AES-CBC', iv: iv.buffer}, key, plaintext + .then (ciphertext) -> + iv_ciphertext = new Uint8Array(ciphertext.byteLength + iv.byteLength) + iv_ciphertext.set iv, 0 + iv_ciphertext.set new Uint8Array(ciphertext), iv.byteLength + + window.crypto.subtle.digest 'SHA-256', iv_ciphertext + .then (digest) -> + computed_sha256 = digest + + window.crypto.subtle.exportKey 'raw', key + .then (key_bytes) -> + resolve [key_bytes, computed_sha256, iv_ciphertext.buffer] + .catch (error) -> + reject error + + ### + @param key_bytes [ArrayBuffer] AES key used for encryption + @param computed_sha256 [ArrayBuffer] SHA-256 checksum of the ciphertext + @param ciphertext [ArrayBuffer] Encrypted plaintext + + @param [ArrayBuffer] + ### + decrypt_aes_asset: (ciphertext, key_bytes, reference_sha256) -> + return new Promise (resolve, reject) -> + window.crypto.subtle.digest 'SHA-256', ciphertext + .then (computed_sha256) -> + a = new Uint32Array reference_sha256 + b = new Uint32Array computed_sha256 + + if not a.every((x, i) -> x is b[i]) + throw new Error 'Encrypted asset does not match its SHA-256 hash' + + window.crypto.subtle.importKey 'raw', key_bytes, 'AES-CBC', false, ['decrypt'] + .then (key) -> + iv = ciphertext.slice 0, 16 + img_ciphertext = ciphertext.slice 16 + window.crypto.subtle.decrypt {name: 'AES-CBC', iv: iv}, key, img_ciphertext + .then (img_plaintext) -> + resolve img_plaintext + .catch (error) -> + reject error diff --git a/app/script/assets/AssetRemoteData.coffee b/app/script/assets/AssetRemoteData.coffee new file mode 100644 index 00000000000..5c1c7042f2e --- /dev/null +++ b/app/script/assets/AssetRemoteData.coffee @@ -0,0 +1,80 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +class z.assets.AssetRemoteData + + ### + Use either z.assets.AssetRemoteData.v2 or z.assets.AssetRemoteData.v3 + to initialize. + + @param otr_key [Uint8Array] + @param sha256 [Uint8Array] + ### + constructor: (@otr_key, @sha256) -> + @download_progress = ko.observable() + @cancel_download = undefined + @generate_url = undefined + + ### + Static initializer for v3 assets + + @param asset_key [String] + @param otr_key [Uint8Array] + @param sha256 [Uint8Array] + @param asset_token [String] token is optional + ### + @v3: (asset_key, otr_key, sha256, asset_token) -> + remote_data = new z.assets.AssetRemoteData otr_key, sha256 + remote_data.generate_url = -> wire.app.service.asset.generate_asset_url_v3 asset_key, asset_token + return remote_data + + ### + Static initializer for v2 assets + + @param conversation_id [String] + @param asset_id [String] + @param otr_key [Uint8Array] + @param sha256 [Uint8Array] + ### + @v2: (conversation_id, asset_id, otr_key, sha256) -> + remote_data = new z.assets.AssetRemoteData otr_key, sha256 + remote_data.generate_url = -> wire.app.service.asset.generate_asset_url_v2 asset_id, conversation_id + return remote_data + + ### + Loads and decrypts stored asset + + @returns [Blob] + ### + load: => + type = undefined + + @_load_buffer() + .then (data) => + [buffer, type] = data + return z.assets.AssetCrypto.decrypt_aes_asset buffer, @otr_key.buffer, @sha256.buffer + .then (buffer) -> + return new Blob [new Uint8Array buffer], type: type + + _load_buffer: => + z.util.load_url_buffer @generate_url(), (xhr) => + xhr.onprogress = (event) => @download_progress Math.round event.loaded / event.total * 100 + @cancel_download = -> xhr.abort.call xhr diff --git a/app/script/assets/AssetRetentionPolicy.coffee b/app/script/assets/AssetRetentionPolicy.coffee new file mode 100644 index 00000000000..dd90f4bb382 --- /dev/null +++ b/app/script/assets/AssetRetentionPolicy.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +z.assets.AssetRetentionPolicy = + ETERNAL: 'eternal' + PERSISTENT: 'persistent' + VOLATILE: 'volatile' diff --git a/app/script/assets/AssetService.coffee b/app/script/assets/AssetService.coffee new file mode 100644 index 00000000000..d63fbaa5d83 --- /dev/null +++ b/app/script/assets/AssetService.coffee @@ -0,0 +1,418 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# AssetService for all asset handling and the calls to the backend REST API. +class z.assets.AssetService + ### + Construct a new Asset Service. + + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.assets.AssetService', z.config.LOGGER.OPTIONS + @rotator = new zeta.webapp.module.image.rotation.ImageFileRotator() + @compressor = new zeta.webapp.module.image.ImageCompressor() + + @BOUNDARY = 'frontier' + + @PREVIEW_CONFIG = + squared: false + max_image_size: 30 + max_byte_size: 1024 + lossy_scaling: true + compression: 0 + + @SMALL_PROFILE_CONFIG = + squared: true + max_image_size: 280 + max_byte_size: 1024 * 1024 + lossy_scaling: true + compression: 0.7 + + @pending_uploads = {} + + ############################################################################### + # REST API calls + ############################################################################### + + ### + Upload any asset to the backend using asset api v1. + + @deprecated + @param config [Object] Configuration object containing the jQuery call settings + @option config [String] url + @option config [Object] data + @option config [String] contentType + @option config [String] contentDisposition + @option config [Function] callback + ### + post_asset: (config) -> + @client.send_request + type: 'POST' + url: config.url + data: config.data + processData: false # otherwise jquery will convert it to a query string + contentType: config.contentType + headers: + 'Content-Disposition': config.contentDisposition + callback: config.callback + + ### + Upload any asset pair to the backend using asset api v1. + + @deprecated + @param small [z.assets.Asset] Small asset for upload + @param medium [z.assets.Asset] Medium asset for upload + ### + post_asset_pair: (small, medium) -> + Promise.all [ + @post_asset + contentType: small.content_type + url: @client.create_url '/assets' + contentDisposition: small.get_content_disposition() + data: small.array_buffer + @post_asset + contentType: medium.content_type + url: @client.create_url '/assets' + contentDisposition: medium.get_content_disposition() + data: medium.array_buffer + ] + + ############################################################################### + # Asset service interactions + ############################################################################### + + ### + Update the user profile image by first making it usable, transforming it and then uploading the asset pair. + + @deprecated + @param conversation_id [String] ID of self conversation + @param file [File, Blob] Image + ### + upload_profile_image: (conversation_id, file, callback) -> + @_convert_image file, (image) => + @_upload_profile_assets conversation_id, image, callback + + ### + Upload arbitrary binary data using the new asset api v3. + The data is AES encrypted before uploading. + + @param bytes [Uint8Array] asset binary data + @param options [Object] + @option public [Boolean] + @option retention [z.assets.AssetRetentionPolicy] + ### + _upload_asset: (bytes, options) -> + key_bytes = null + sha256 = null + + z.assets.AssetCrypto.encrypt_aes_asset bytes + .then (data) => + [key_bytes, sha256, ciphertext] = data + return @post_asset_v3 ciphertext, options + .then (data) -> + {key, token} = data + return [key_bytes, sha256, key, token] + + ### + Upload image the new asset api v3. Promise will resolve with z.proto.Asset instance. + In case of an successful upload the uploaded property is set. Otherwise it will be marked as not + uploaded. + + @param file [File, Blob] Image + @param options [Object] + @option public [Boolean] + @option retention [z.assets.AssetRetentionPolicy] + ### + upload_image_asset: (file, options) -> + compressed_image = null + image_bytes = null + + @compress_image file + .then (data) -> + [original_image, compressed_image] = data + return z.util.base64_to_array compressed_image.src + .then (bytes) => + image_bytes = bytes + @_upload_asset image_bytes, options + .then (data) -> + [key_bytes, sha256, key, token] = data + image_meta_data = new z.proto.Asset.ImageMetaData compressed_image.width, compressed_image.height + asset = new z.proto.Asset() + asset.set 'original', new z.proto.Asset.Original file.type, image_bytes.length, null, image_meta_data + asset.set 'uploaded', new z.proto.Asset.RemoteData key_bytes, sha256, key, token + return asset + .catch (error) => + @logger.log @logger.levels.ERROR, error + asset = new z.proto.Asset() + asset.set 'not_uploaded', z.proto.Asset.NotUploaded.FAILED + return asset + + ### + Generates the URL an asset can be downloaded from. + + @param asset_id [String] ID of the asset + @param conversation_id [String] ID of the conversation the asset belongs to + @return [String] Asset URL + ### + generate_asset_url: (asset_id, conversation_id) -> + url = @client.create_url "/assets/#{asset_id}" + asset_url = "#{url}?access_token=#{@client.access_token}&conv_id=#{conversation_id}" + return asset_url + + ### + Generates the URL for asset api v2. + + @param asset_id [String] ID of the asset + @param conversation_id [String] ID of the conversation the asset belongs to + @return [String] Asset URL + ### + generate_asset_url_v2: (asset_id, conversation_id) -> + url = @client.create_url "/conversations/#{conversation_id}/otr/assets/#{asset_id}" + asset_url = "#{url}?access_token=#{@client.access_token}" + return asset_url + + ### + Generates the URL for asset api v3. + + @param asset_key [String] + @param asset_token [String] + @return [String] Asset URL + ### + generate_asset_url_v3: (asset_key, asset_token) -> + url = @client.create_url "/assets/v3/#{asset_key}/" + asset_url = "#{url}?access_token=#{@client.access_token}" + asset_url = "#{asset_url}&asset_token=#{asset_token}" if asset_token + return asset_url + + ############################################################################### + # Private + ############################################################################### + + ### + Compress image before uploading. + + @param file [File, Blob] Image + ### + compress_image: (file) -> + return new Promise (resolve) => + @_convert_image file, (image) => + @compressor.transform_image image, (compressed_image) -> + resolve [image, compressed_image] + + ### + Convert an image before uploading it. + + @param file [File, Blob] Image + @param callback [Function] Function to be called on return + ### + _convert_image: (file, callback) -> + @_rotate_image file, (rotated_file) -> + z.util.read_deferred(rotated_file, 'url').done (url) -> + image = new Image() + image.onload = -> callback image + image.onerror = (e) => @logger.log "Loading image failed #{e}" + image.src = url + ### + Rotate an image file unless it is a gif. + + @private + + @param file [Object] Image file to be rotated + @param callback [Function] Function to be called on return + ### + _rotate_image: (file, callback) -> + return callback file if file.type is 'image/gif' + @rotator.rotate file, callback + + ### + Update the profile image of the user. + + @note We need to upload it in sizes 'smallProfile', and 'medium'. + @private + + @param conversation_id [String] ID of the self conversation + @param image [Object] Image to be used as new profile picture + @param callback [Function] Function to be called on server return + ### + _upload_profile_assets: (conversation_id, image, callback) -> + # Finished compressing medium image + medium_image_compressed = (medium_image) => + medium_asset = new z.assets.Asset + conversation_id: conversation_id + original_width: medium_image.width + original_height: medium_image.height + public: true + medium_asset.set_image medium_image + + # Finished compressing small image + small_profile_image_compressed = (small_image) => + small_profile_asset = $.extend true, {}, medium_asset + small_profile_asset.payload.tag = z.assets.ImageSizeType.SMALL_PROFILE + small_profile_asset.set_image small_image + + @post_asset_pair small_profile_asset, medium_asset + .then (value) -> + [small_response, medium_response] = value + callback [small_response.data, medium_response.data] + .catch (error) -> + callback [], error + + # Compress small image + @compressor.transform_image medium_image, small_profile_image_compressed, @SMALL_PROFILE_CONFIG + + # Compress medium image + @compressor.transform_image image, medium_image_compressed # default config # + + ### + Create request data for asset upload. + + @param asset_data [UInt8Array|ArrayBuffer] Asset data + @param metadata [Object] image meta data + ### + _create_asset_multipart_body: (asset_data, metadata) -> + metadata = JSON.stringify metadata + asset_data_md5 = z.util.encode_base64_md5_array_buffer_view asset_data + + body = '' + body += '--' + @BOUNDARY + '\r\n' + body += 'Content-Type: application/json; charset=utf-8\r\n' + body += "Content-length: #{metadata.length}\r\n" + body += '\r\n' + body += metadata + '\r\n' + body += '--' + @BOUNDARY + '\r\n' + body += 'Content-Type: application/octet-stream\r\n' + body += "Content-length: #{asset_data.length}\r\n" + body += "Content-MD5: #{asset_data_md5}\r\n" + body += '\r\n' + + footer = '\r\n--' + @BOUNDARY + '--\r\n' + + return new Blob [body, asset_data, footer] + + ### + Post assets to a conversation. + + @deprecated + @param conversation_id [String] ID of the self conversation + @param json_payload [Object] First part of the multipart message + @param body_payload [Object] Image to be used as new profile picture + @param force_sending [Boolean] Force sending + @param upload_id [String] Identifies the upload request + ### + post_asset_v2: (conversation_id, json_payload, body_payload, force_sending, upload_id) -> + return new Promise (resolve, reject) => + url = @client.create_url "/conversations/#{conversation_id}/otr/assets" + url = "#{url}?ignore_missing=true" if force_sending + + data = @_create_asset_multipart_body body_payload, json_payload + pending_uploads = @pending_uploads + + xhr = new XMLHttpRequest() + xhr.open 'POST', url + xhr.setRequestHeader 'Content-Type', 'multipart/mixed; boundary=' + @BOUNDARY + xhr.setRequestHeader 'Authorization', "#{@client.access_token_type} #{@client.access_token}" + xhr.onload = (event) -> + if @status is 201 + resolve [JSON.parse(@response), @getResponseHeader 'Location'] + else if @status is 412 + reject JSON.parse @response + else + reject event + delete pending_uploads[upload_id] + xhr.onerror = (error) -> + reject error + delete pending_uploads[upload_id] + xhr.upload.onprogress = (event) -> + if upload_id + # we use amplify due to the fact that Promise API lacks progress support + percentage_progress = Math.round(event.loaded / event.total * 100) + amplify.publish 'upload' + upload_id, percentage_progress + xhr.send data + + pending_uploads[upload_id] = xhr + + ### + Post assets using asset api v3. + + @param asset_data [Uint8Array|ArrayBuffer] + @param metadata [Object] + @option public [Boolean] Default is false + @option retention [z.assets.AssetRetentionPolicy] Default is z.assets.AssetRetentionPolicy.PERSISTENT + @param xhr_accessor_function [Function] Function will get a reference to the underlying XMLHTTPRequest + ### + post_asset_v3: (asset_data, metadata, xhr_accessor_function) -> + return new Promise (resolve, reject) => + metadata = $.extend + public: false + retention: z.assets.AssetRetentionPolicy.PERSISTENT + , metadata + + xhr = new XMLHttpRequest() + xhr.open 'POST', @client.create_url "/assets/v3" + xhr.setRequestHeader 'Content-Type', 'multipart/mixed; boundary=' + @BOUNDARY + xhr.setRequestHeader 'Authorization', "#{@client.access_token_type} #{@client.access_token}" + xhr.onload = (event) -> if @status is 201 then resolve JSON.parse(@response) else reject event + xhr.onerror = reject + xhr_accessor_function? xhr + xhr.send @_create_asset_multipart_body new Uint8Array(asset_data), metadata + + ### + Cancel an asset upload. + + @param upload_id [String] Identifies the upload request + ### + cancel_asset_upload: (upload_id) => + xhr = @pending_uploads[upload_id] + if xhr? + xhr.abort() + delete @pending_uploads[upload_id] + + ### + Post an OTR asset to a conversation. + + @param file [File, Blob] Image + @param image [Object] Image to be used as new profile picture + ### + create_asset_proto: (file) -> + original_image = null + compressed_image = null + image_bytes = null + + @compress_image file + .then (data) -> + [original_image, compressed_image] = data + return z.util.base64_to_array compressed_image.src + .then (data) -> + image_bytes = data + z.assets.AssetCrypto.encrypt_aes_asset image_bytes + .then ([key_bytes, sha256, ciphertext]) -> + image_asset = new z.proto.ImageAsset() + image_asset.set_tag z.assets.ImageSizeType.MEDIUM + image_asset.set_width compressed_image.width + image_asset.set_height compressed_image.height + image_asset.set_original_width original_image.width + image_asset.set_original_height original_image.height + image_asset.set_mime_type file.type + image_asset.set_size image_bytes.length + image_asset.set_otr_key key_bytes + image_asset.set_sha256 sha256 + return [image_asset, new Uint8Array ciphertext] diff --git a/app/script/assets/AssetTransferState.coffee b/app/script/assets/AssetTransferState.coffee new file mode 100644 index 00000000000..ed90b5bd2fb --- /dev/null +++ b/app/script/assets/AssetTransferState.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# Enum of different asset upload status. +z.assets.AssetTransferState = + UPLOADING: 'uploading' + UPLOADED: 'uploaded' + UPLOAD_FAILED: 'upload-failed' + UPLOAD_CANCELED: 'upload-canceled' + DOWNLOADING: 'downloading' diff --git a/app/script/assets/AssetType.coffee b/app/script/assets/AssetType.coffee new file mode 100644 index 00000000000..827d0b7ef42 --- /dev/null +++ b/app/script/assets/AssetType.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# Enum of different asset types. +z.assets.AssetType = + FILE: 'File' + LOCATION: 'Location' + MEDIUM_IMAGE: 'MediumImage' + PREVIEW_IMAGE: 'PreviewImage' + TEXT: 'Text' diff --git a/app/script/assets/AssetUploadFailedReason.coffee b/app/script/assets/AssetUploadFailedReason.coffee new file mode 100644 index 00000000000..61e514bcf1c --- /dev/null +++ b/app/script/assets/AssetUploadFailedReason.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# Enum of different asset upload status. +z.assets.AssetUploadFailedReason = + FAILED: 1 + CANCELLED: 0 diff --git a/app/script/assets/ImageSizeType.coffee b/app/script/assets/ImageSizeType.coffee new file mode 100644 index 00000000000..119f335cbc8 --- /dev/null +++ b/app/script/assets/ImageSizeType.coffee @@ -0,0 +1,26 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.assets ?= {} + +# Enum of different image size types. +z.assets.ImageSizeType = + MEDIUM: 'medium' + PREVIEW: 'preview' + SMALL_PROFILE: 'smallProfile' diff --git a/app/script/audio/AudioPlayingType.coffee b/app/script/audio/AudioPlayingType.coffee new file mode 100644 index 00000000000..09699e589bf --- /dev/null +++ b/app/script/audio/AudioPlayingType.coffee @@ -0,0 +1,40 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.audio ?= {} + +# Enum of sounds playing for different sound settings. +z.audio.AudioPlayingType = + NONE: [ + z.audio.AudioType.CALL_DROP + z.audio.AudioType.NETWORK_INTERRUPTION + z.audio.AudioType.OUTGOING_CALL + z.audio.AudioType.READY_TO_TALK + z.audio.AudioType.TALK_LATER + ] + SOME: [ + z.audio.AudioType.CALL_DROP + z.audio.AudioType.INCOMING_CALL + z.audio.AudioType.INCOMING_PING + z.audio.AudioType.NETWORK_INTERRUPTION + z.audio.AudioType.OUTGOING_CALL + z.audio.AudioType.OUTGOING_PING + z.audio.AudioType.READY_TO_TALK + z.audio.AudioType.TALK_LATER + ] diff --git a/app/script/audio/AudioRepository.coffee b/app/script/audio/AudioRepository.coffee new file mode 100644 index 00000000000..5f96bd9a049 --- /dev/null +++ b/app/script/audio/AudioRepository.coffee @@ -0,0 +1,160 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.audio ?= {} + +# Enum of audio settings. +z.audio.AudioSetting = + ALL: 'all' + NONE: 'none' + SOME: 'some' + +AUDIO_PATH = '/audio' + +# Audio repository for all audio interactions. +class z.audio.AudioRepository + # Construct a new Audio Repository. + constructor: -> + @logger = new z.util.Logger 'z.audio.AudioRepository', z.config.LOGGER.OPTIONS + @audio_context = undefined + @in_loop = {} + @_init_sound_manager() + @_subscribe_to_audio_properties() + + # Closing the AudioContext. + close_audio_context: => + if @audio_context + @audio_context.close() + @audio_context = undefined + @logger.log @logger.levels.INFO, 'Closed existing AudioContext' + + # Initialize the AudioContext. + get_audio_context: => + if @audio_context + @logger.log @logger.levels.INFO, 'Reusing existing AudioContext', @audio_context + return @audio_context + else if window.AudioContext + @audio_context = new window.AudioContext() + @logger.log @logger.levels.INFO, 'Initialized a new AudioContext', @audio_context + return @audio_context + else + @logger.log @logger.levels.ERROR, 'The flow audio cannot use the Web Audio API as it is unavailable.' + return undefined + + ### + Initialize a sound. + + @private + @param id [z.audio.AudioType] ID of the sound + @param url [String] URL of sound file + @return [Function] Function to set up the sound in SoundManager + ### + _init_sound: (id, url) -> + return soundManager.createSound id: id, url: url + + # Initialize all sounds. + _init_sounds: -> + @alert = @_init_sound z.audio.AudioType.ALERT, "#{AUDIO_PATH}/alert.mp3" + @call_drop = @_init_sound z.audio.AudioType.CALL_DROP, "#{AUDIO_PATH}/call_drop.mp3" + @network_interruption = @_init_sound z.audio.AudioType.NETWORK_INTERRUPTION, "#{AUDIO_PATH}/nw_interruption.mp3" + @new_message = @_init_sound z.audio.AudioType.NEW_MESSAGE, "#{AUDIO_PATH}/new_message.mp3" + @ping_from_me = @_init_sound z.audio.AudioType.OUTGOING_PING, "#{AUDIO_PATH}/ping_from_me.mp3" + @ping_from_them = @_init_sound z.audio.AudioType.INCOMING_PING, "#{AUDIO_PATH}/ping_from_them.mp3" + @ready_to_talk = @_init_sound z.audio.AudioType.READY_TO_TALK, "#{AUDIO_PATH}/ready_to_talk.mp3" + @ringing_from_me = @_init_sound z.audio.AudioType.OUTGOING_CALL, "#{AUDIO_PATH}/ringing_from_me.mp3" + @ringing_from_them = @_init_sound z.audio.AudioType.INCOMING_CALL, "#{AUDIO_PATH}/ringing_from_them.mp3" + @talk_later = @_init_sound z.audio.AudioType.TALK_LATER, "#{AUDIO_PATH}/talk_later.mp3" + + # Use Amplify to subscribe to all audio playback related events. + _subscribe_to_audio_events: -> + amplify.subscribe z.event.WebApp.AUDIO.PLAY, @, @_play + amplify.subscribe z.event.WebApp.AUDIO.PLAY_IN_LOOP, @, @_play_in_loop + amplify.subscribe z.event.WebApp.AUDIO.STOP, @, @_stop + + # Use Amplify to subscribe to all audio properties related events. + _subscribe_to_audio_properties: -> + @sound_setting = ko.observable z.audio.AudioSetting.ALL + @sound_setting.subscribe (sound_setting) => + @_stop_all() if sound_setting is z.audio.AudioSetting.NONE + + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATED, (properties) => + @sound_setting properties.settings.sound.alerts + + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATE.SOUND_ALERTS, (value) => + @sound_setting value + + # Initialize the SoundManager. + _init_sound_manager: -> + soundManager.setup + debugMode: false + useConsole: false + onready: => + @_init_sounds() + @_subscribe_to_audio_events() + + ### + Start playback of a sound + @param audio_id [String] Sound that should be played + ### + _play: (audio_id) -> + audio = soundManager.getSoundById audio_id + + return if @sound_setting() is z.audio.AudioSetting.NONE and audio_id not in z.audio.AudioPlayingType.NONE + return if @sound_setting() is z.audio.AudioSetting.SOME and audio_id not in z.audio.AudioPlayingType.SOME + + @logger.log "Playing sound: #{audio_id}", audio + audio.play() + + ### + Start playback of a sound in a loop. + + @note Prevent playing multiples instances of looping sounds + @param audio [Object] SoundManager sound object + @param is_first_time [Boolean] Is this the initial call or an on finish loop + ### + _play_in_loop: (audio_id, is_first_time = true) -> + audio = soundManager.getSoundById audio_id + + return if @sound_setting() is z.audio.AudioSetting.NONE and audio_id not in z.audio.AudioPlayingType.NONE + return if @sound_setting() is z.audio.AudioSetting.SOME and audio_id not in z.audio.AudioPlayingType.SOME + + if not @in_loop[audio_id] + @logger.log "Looping sound: #{audio_id}", audio + @in_loop[audio_id] = audio_id + else + return if is_first_time + + audio.play onfinish: => + @_play_in_loop audio.id, false + + ### + Stop playback of a sound. + @param audio [Object] SoundManager sound object + ### + _stop: (audio_id) -> + audio = soundManager.getSoundById audio_id + + @logger.log "Stopping sound: #{audio_id}", audio + audio.stop() + + delete @in_loop[audio_id] if @in_loop[audio_id] + + # Stop all sounds playing in loop. + _stop_all: -> + @_stop sound for sound of @in_loop diff --git a/app/script/audio/AudioType.coffee b/app/script/audio/AudioType.coffee new file mode 100644 index 00000000000..1596ff4e7e8 --- /dev/null +++ b/app/script/audio/AudioType.coffee @@ -0,0 +1,33 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.audio ?= {} + +# Enum of different supported sounds. +z.audio.AudioType = + ALERT: 'alert' + CALL_DROP: 'call-drop' + INCOMING_CALL: 'ringing-from-them' + INCOMING_PING: 'ping-from-them' + NETWORK_INTERRUPTION: 'network-interruption' + NEW_MESSAGE: 'new-message' + OUTGOING_CALL: 'ringing-from-me' + OUTGOING_PING: 'ping-from-me' + READY_TO_TALK: 'ready-to-talk' + TALK_LATER: 'talk-later' diff --git a/app/script/auth/AccessTokenError.coffee b/app/script/auth/AccessTokenError.coffee new file mode 100644 index 00000000000..72c5332af6c --- /dev/null +++ b/app/script/auth/AccessTokenError.coffee @@ -0,0 +1,36 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +class z.auth.AccessTokenError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @type = type + @stack = (new Error()).stack + + @:: = new Error() + @::constructor = @ + @::TYPE = { + NOT_FOUND_IN_CACHE: 'z.auth.AccessTokenError::TYPE.NOT_FOUND_IN_CACHE' + REFRESH_IN_PROGRESS: 'z.auth.AccessTokenError::TYPE.REFRESH_IN_PROGRESS' + REQUEST_FAILED: 'z.auth.AccessTokenError::TYPE.REQUEST_FAILED' + REQUEST_FORBIDDEN: 'z.auth.AccessTokenError::TYPE.REQUEST_FORBIDDEN' + } diff --git a/app/script/auth/AuthRepository.coffee b/app/script/auth/AuthRepository.coffee new file mode 100644 index 00000000000..afac30f2506 --- /dev/null +++ b/app/script/auth/AuthRepository.coffee @@ -0,0 +1,255 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +# Authentication Repository for all authentication and registration interactions with the authentication service. +class z.auth.AuthRepository + ### + Construct a new Authentication Repository. + @param auth_service [z.auth.AuthService] Backend REST API service implementation + ### + constructor: (@auth_service) -> + @logger = new z.util.Logger 'z.auth.AuthRepository', z.config.LOGGER.OPTIONS + + @access_token_refresh = undefined + + amplify.subscribe z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEW, @renew_access_token + + # Print all cookies for a user in the console. + list_cookies: -> + @auth_service.get_cookies + .then (cookies) => + @logger.force_log 'Backend cookies:' + for cookie, index in cookies + expiration = z.util.format_timestamp cookie.time + log = "Label: #{cookie.label} | Type: #{cookie.type} | Expiration: #{expiration}" + @logger.force_log "Cookie No. #{index + 1} | #{log}" + .catch (error) => + @logger.force_log 'Could not list user cookies', error + + ### + Login (with email or phone) in order to obtain an access-token and cookie. + + @param login [Object] Containing sign in information + @option login [String] email The email address for a password login + @option login [String] phone The phone number for a password or SMS login + @option login [String] password The password for a password login + @option login [String] code The login code for an SMS login + @param persist [Boolean] Request a persistent cookie instead of a session cookie + @return [Promise] Promise that resolves with the received access token + ### + login: (login, persist) => + return new Promise (resolve, reject) => + @auth_service.post_login login, persist + .then (response) => + @save_access_token response + z.storage.set_value z.storage.StorageKey.AUTH.PERSIST, persist + z.storage.set_value z.storage.StorageKey.AUTH.SHOW_LOGIN, true + resolve response + .catch (error) -> reject error + + ### + Logout the user on the backend. + @return [Promise] Promise that will always resolve + ### + logout: => + return new Promise (resolve) => + @auth_service.post_logout() + .then => + @logger.log @logger.levels.INFO, 'Log out on backend successful' + resolve() + .catch (error) => + @logger.log @logger.levels.WARN, "Log out on backend failed: #{error.message}", error + resolve() + + ### + Register a new user (with email). + + @param new_user [Object] Containing the email, username and password needed for account creation + @option new_user [String] name + @option new_user [String] email + @option new_user [String] password + @option new_user [String] label Cookie label + @return [Promise] Promise that will resolve on success + ### + register: (new_user) => + return new Promise (resolve, reject) => + @auth_service.post_register new_user + .then (response) => + z.storage.set_value z.storage.StorageKey.AUTH.PERSIST, true + z.storage.set_value z.storage.StorageKey.AUTH.SHOW_LOGIN, true + z.storage.set_value new_user.label_key, new_user.label + @logger.log @logger.levels.INFO, + "COOKIE::'#{new_user.label}' Saved cookie label with key '#{new_user.label_key}' in Local Storage", { + key: new_user.label_key, + value: new_user.label + } + resolve response + .catch (error) -> reject error + + ### + Resend an email or phone activation code. + + @param send_activation_code [Object] Containing the email or phone number needed to resend activation email + @option send_activation_code [String] email + @return [Promise] Promise that resolves on success + ### + resend_activation: (send_activation_code) => + @auth_service.post_activate_send send_activation_code + + ### + Retrieve personal invite information. + @param invite [String] Invite code + @return [Promise] Promise that resolves with the invite data + ### + retrieve_invite: (invite) => + @auth_service.get_invitations_info invite + + ### + Request SMS validation code. + @param request_code [Object] Containing the phone number in E.164 format and whether a code should be forced + @return [Promise] Promise that resolve on success + ### + request_login_code: (request_code) => + @auth_service.post_login_send request_code + + ### + Renew access-token provided a valid cookie. + ### + renew_access_token: => + @get_access_token() + .then => + @logger.log @logger.levels.INFO, 'Refreshed Access Token successfully.' + amplify.publish z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEWED + .catch (error) => + if error.type is z.auth.AccessTokenError::TYPE.REQUEST_FORBIDDEN + @logger.log @logger.levels.WARN, "Session expired on access token refresh: #{error.message}", error + Raygun.send error + amplify.publish z.event.WebApp.SIGN_OUT, 'session_expired', false, true + else if error.type isnt z.auth.AccessTokenError::TYPE.REFRESH_IN_PROGRESS + @logger.log @logger.levels.ERROR, "Refreshing access token failed: '#{error.type}'", error + # @todo What do we do in this case? + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + + # Get the cached access token from the Amplify store. + get_cached_access_token: -> + return new Promise (resolve, reject) => + access_token = z.storage.get_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.VALUE + access_token_type = z.storage.get_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.TYPE + + if access_token + @logger.log @logger.levels.INFO, 'Cached access token found in Local Storage', access_token + @auth_service.save_access_token_in_client access_token_type, access_token + @_schedule_token_refresh z.storage.get_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.EXPIRATION + resolve() + else + error_message = 'No cached access token found in Local Storage' + reject new z.auth.AccessTokenError error_message, z.auth.AccessTokenError::TYPE.NOT_FOUND_IN_CACHE + + ### + Initially get access-token provided a valid cookie. + @return [Promise] Returns a Promise that resolve with the access token data + ### + get_access_token: => + return new Promise (resolve, reject) => + if @auth_service.client.is_requesting_access_token() + error_message = 'Access Token request already in progress' + @logger.log @logger.levels.WARN, error_message + reject new z.auth.AccessTokenError error_message, z.auth.AccessTokenError::TYPE.REFRESH_IN_PROGRESS + else + @auth_service.post_access() + .then (access_token) => + @save_access_token access_token + resolve access_token + .catch (error) -> + reject error + + ### + Store the access token using Amplify. + + @example Access Token data we expect: + access_token: Lt-IRHxkY9JLA5UuBR3Exxj5lCUf... + access_token_expiration: 1424951321067 => Thu, 26 Feb 2015 11:48:41 GMT + access_token_type: Bearer + access_token_ttl: 900000 => 900s/15min + + @param access_token_data [Object, String] Access Token + @option access_token_data [String] access_token + @option access_token_data [String] expires_in + @option access_token_data [String] type + ### + save_access_token: (access_token_data) => + expires_in_millis = 1000 * access_token_data.expires_in + expiration_timestamp = Date.now() + expires_in_millis + + z.storage.set_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.VALUE, access_token_data.access_token, access_token_data.expires_in + z.storage.set_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.EXPIRATION, expiration_timestamp, access_token_data.expires_in + z.storage.set_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.TTL, expires_in_millis, access_token_data.expires_in + z.storage.set_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.TYPE, access_token_data.token_type, access_token_data.expires_in + + @logger.log @logger.levels.LEVEL_1, 'Saved access token.', access_token_data + @_log_access_token_expiration expiration_timestamp + @_schedule_token_refresh expiration_timestamp + + @auth_service.save_access_token_in_client access_token_data.token_type, access_token_data.access_token + + # Deletes all access token data stored on the client. + delete_access_token: -> + z.storage.reset_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.VALUE + z.storage.reset_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.EXPIRATION + z.storage.reset_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.TTL + z.storage.reset_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.TYPE + + ### + Logs the expiration time of the access token. + @private + @param expiration_timestamp [Integer] Timestamp when access token expires + ### + _log_access_token_expiration: (expiration_timestamp) => + expiration_log = z.util.format_timestamp expiration_timestamp + @logger.log @logger.levels.INFO, "Your access token will expire on: #{expiration_log}" + + ### + Refreshes the access token in time before it expires. + + @note Access token will be refreshed 1 minute (60000ms) before it expires + @private + @param expiration_timestamp [Integer] The expiration date (and time) as timestamp + ### + _schedule_token_refresh: (expiration_timestamp) => + window.clearTimeout @access_token_refresh if @access_token_refresh + callback_timestamp = expiration_timestamp - 60000 + + if callback_timestamp < Date.now() + @logger.log @logger.levels.INFO, 'Immediately executing access token refresh' + @renew_access_token() + else + time = z.util.format_timestamp callback_timestamp + @logger.log @logger.levels.INFO, "Scheduling next access token refresh for '#{time}'" + + @access_token_refresh = window.setTimeout => + if callback_timestamp > (Date.now() + 15000) + @logger.log @logger.levels.INFO, "Access token refresh scheduled for '#{time}' skipped because it was executed late" + else if navigator.onLine + @logger.log @logger.levels.INFO, "Access token refresh scheduled for '#{time}' executed" + @renew_access_token() + else + @logger.log @logger.levels.INFO, "Access token refresh scheduled for '#{time}' skipped because we are offline" + , callback_timestamp - Date.now() diff --git a/app/script/auth/AuthService.coffee b/app/script/auth/AuthService.coffee new file mode 100644 index 00000000000..4ce015678ee --- /dev/null +++ b/app/script/auth/AuthService.coffee @@ -0,0 +1,303 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +# Authentication Service for all authentication and registration calls to the backend REST API. +class z.auth.AuthService + URL_ACCESS: '/access' + URL_ACTIVATE: '/activate' + URL_COOKIES: '/cookies' + URL_INVITATIONS: '/invitations' + URL_LOGIN: '/login' + URL_REGISTER: '/register' + + ### + Construct a new Authentication Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.auth.AuthService', z.config.LOGGER.OPTIONS + + + ############################################################################### + # Authentication + ############################################################################### + + ### + Get all cookies for a user. + @return [Promise] Promise that resolves with an array of cookies + ### + get_cookies: -> + return new Promise (resolve, reject) => + $.ajax + crossDomain: true + headers: + Authorization: "Bearer #{window.decodeURIComponent(@client.access_token)}" + type: 'GET' + url: @client.create_url "#{@URL_COOKIES}" + .done (data) -> + resolve data.cookies + .fail (jqXHR, textStatus, errorThrown) -> + reject jqXHR.responseJSON or errorThrown + + ### + Get invite information. + @param code [String] Invite code + @return [Promise] Promise that resolves with invitations information + ### + get_invitations_info: (code) -> + return new Promise (resolve, reject) => + $.ajax + crossDomain: true + type: 'GET' + url: @client.create_url "#{@URL_INVITATIONS}/info" + data: "code=#{code}" + .done (data) -> + resolve data + .fail (jqXHR, textStatus, errorThrown) -> + reject jqXHR.responseJSON or errorThrown + + ### + Get access-token if a valid cookie is provided. + + @note Don't use our client wrapper here, because to query "/access" we need to set "withCredentials" to "true" in order to send the cookie. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/auth/authenticate + + @param options [Object] + @option options [Boolean] should_retry Should we retry on error + @option options [Boolean] retry_limit Should we retry on error + ### + post_access: (options) -> + return new Promise (resolve, reject) => + @client.is_requesting_access_token true + + options = $.extend + should_retry: true + retries: 0 + retry_limit: 10 + , options + + settings = + crossDomain: true + type: 'POST' + url: @client.create_url "#{@URL_ACCESS}" + xhrFields: + withCredentials: true + success: (data) => + @logger.log @logger.levels.INFO, + "Requesting access token successful after #{options.retries + 1} attempt(s)", data + @save_access_token_in_client data.token_type, data.access_token + resolve data + error: (jqXHR, textStatus, errorThrown) => + options.retries++ + should_retry = options.should_retry and options.retries < options.retry_limit + if not navigator.onLine + @logger.log @logger.levels.WARN, 'Access token refresh paused due to lack of internet connectivity' + $(window).on 'online', => + @logger.log @logger.levels.INFO, 'Internet connectivity regained. Continuing access token refresh.' + $.ajax settings + else if should_retry and jqXHR.status isnt z.service.BackendClientError::STATUS_CODE.FORBIDDEN + window.setTimeout => + @logger.log @logger.levels.INFO, + "Trying to get a new access token - attempt '#{options.retries}'" + $.ajax settings + , 500 + else + error_description = "Requesting access token failed after '#{options.retries}' attempt(s): #{errorThrown}" + if jqXHR.responseJSON or jqXHR.responseText?.startsWith '{' + error = jqXHR.responseJSON or JSON.parse jqXHR.responseText + if error.code is z.service.BackendClientError::STATUS_CODE.FORBIDDEN + error_message = "#{error_description} (#{error.label} - #{error.message})" + error = new z.auth.AccessTokenError error_message, z.auth.AccessTokenError::TYPE.REQUEST_FORBIDDEN + else + error_message = 'Access token refresh failed' + error = new z.auth.AccessTokenError error_message, z.auth.AccessTokenError::TYPE.REQUEST_FAILED + Raygun.send error + + @save_access_token_in_client() + @logger.log @logger.levels.ERROR, error_description, jqXHR + reject error + + if @client.access_token + settings.headers = + Authorization: "Bearer #{window.decodeURIComponent(@client.access_token)}" + + $.ajax settings + + ### + Resend an email or phone activation code. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/sendActivationCode + + @param send_activation_code [Object] Containing the email or phone number needed to resend activation email + @option send_activation_code [String] email + @return [Promise] Promise that resolves on successful code resend + ### + post_activate_send: (send_activation_code) -> + return new Promise (resolve, reject) => + $.ajax + contentType: 'application/json; charset=utf-8' + crossDomain: true + data: pako.gzip JSON.stringify send_activation_code + headers: + 'Content-Encoding': 'gzip' + processData: false + type: 'POST' + url: @client.create_url "#{@URL_ACTIVATE}/send" + .done -> + resolve z.service.BackendClientError::STATUS_CODE.OK + .fail (jqXHR, textStatus, errorThrown) -> + reject jqXHR.responseJSON or errorThrown + + ### + Delete all cookies on the backend. + + @param email [String] The user's e-mail address + @param password [String] The user's password + @param labels [Array] A list of cookie labels to remove from the system (optional) + ### + post_cookies_remove: (email, password, labels) -> + @client.send_json + url: @client.create_url "#{@URL_COOKIES}/remove" + type: 'POST' + data: + email: email + password: password + labels: labels + + ### + Login in order to obtain an access-token and cookie. + + @note Don't use our client wrapper here. On cookie requests we need to use plain jQuery AJAX. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/auth/login + + @param login [Object] Containing sign in information + @option login [String] email The email address for a password login + @option login [String] phone The phone number for a password or SMS login + @option login [String] password The password for a password login + @option login [String] code The login code for an SMS login + @param persist [Boolean] Request a persistent cookie instead of a session cookie + @return [Promise] Promise that resolves with access token + ### + post_login: (login, persist) -> + return new Promise (resolve, reject) => + $.ajax + contentType: 'application/json; charset=utf-8' + crossDomain: true + data: pako.gzip JSON.stringify login + headers: + 'Content-Encoding': 'gzip' + processData: false + type: 'POST' + url: @client.create_url "#{@URL_LOGIN}?persist=#{persist}" + xhrFields: + withCredentials: true + .done (data) -> + resolve data + .fail (jqXHR, textStatus, errorThrown) => + if jqXHR.status is z.service.BackendClientError::STATUS_CODE.TOO_MANY_REQUESTS and login.email + # Backend blocked our user account from login, so we have to reset our cookies + @post_cookies_remove login.email, login.password, undefined + .then -> reject jqXHR.responseJSON or errorThrown + else + reject jqXHR.responseJSON or errorThrown + + ### + A login code can be used only once and times out after 10 minutes. + + @note Only one login code may be pending at a time. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/sendLoginCode + + @param request_code [Object] Containing the phone number in E.164 format and whether a code should be forced + @return [Promise] Promise that resolves on successful login code request + ### + post_login_send: (request_code) -> + return new Promise (resolve, reject) => + $.ajax + contentType: 'application/json; charset=utf-8' + data: pako.gzip JSON.stringify request_code + headers: + 'Content-Encoding': 'gzip' + processData: false + type: 'POST' + url: @client.create_url "#{@URL_LOGIN}/send" + .done (data) -> + resolve data + .fail (jqXHR, textStatus, errorThrown) -> + reject jqXHR.responseJSON or errorThrown + + ### + Logout on the backend side. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/auth/logout + ### + post_logout: -> + return new Promise (resolve, reject) => + $.ajax + crossDomain: true + headers: + Authorization: "Bearer #{window.decodeURIComponent(@client.access_token)}" + type: 'POST' + url: @client.create_url "#{@URL_ACCESS}/logout" + xhrFields: + withCredentials: true + .done (data) -> + resolve data + .fail (jqXHR, textStatus, errorThrown) -> + reject [jqXHR.responseJSON or errorThrown] + + ### + Register a new user. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/register + + @param new_user [Object] Containing the email, username and password needed for account creation + @option new_user [String] name + @option new_user [String] email + @option new_user [String] password + @option new_user [String] locale + @return [Promise] Promise that will resolve on success + ### + post_register: (new_user) -> + return new Promise (resolve, reject) => + $.ajax + contentType: 'application/json; charset=utf-8' + crossDomain: true + data: pako.gzip JSON.stringify new_user + headers: + 'Content-Encoding': 'gzip' + processData: false + type: 'POST' + url: @client.create_url "#{@URL_REGISTER}?challenge_cookie=true" + xhrFields: + withCredentials: true + .done (data) -> + resolve data + .fail (jqXHR, textStatus, errorThrown) -> + reject jqXHR.responseJSON or errorThrown + + ### + Save the access token date in the client. + @param type [String] Access token type + @param value [String] Access token + ### + save_access_token_in_client: (type = '', value = '') => + @client.access_token_type = type + @client.access_token = value + @client.is_requesting_access_token false diff --git a/app/script/auth/AuthView.coffee b/app/script/auth/AuthView.coffee new file mode 100644 index 00000000000..6694ef37a9e --- /dev/null +++ b/app/script/auth/AuthView.coffee @@ -0,0 +1,62 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +z.auth.AuthView = + ANIMATION_DIRECTION: + HORIZONTAL_LEFT: 'horizontal-left' + HORIZONTAL_RIGHT: 'horizontal-right' + VERTICAL_BOTTOM: 'vertical-bottom' + VERTICAL_TOP: 'vertical-top' + MODE: + ACCOUNT_EMAIL: 'email' + ACCOUNT_LOGIN: 'login' + ACCOUNT_REGISTER: 'register' + ACCOUNT_PHONE: 'phone' + HISTORY: 'history' + LIMIT: 'limit' + POSTED: 'posted' + POSTED_OFFLINE: 'offline' + POSTED_PENDING: 'pending' + POSTED_RESEND: 'resend' + POSTED_RETRY: 'retry' + POSTED_VERIFY: 'verify' + VERIFY_ADD_EMAIL: 'add_email' + VERIFY_CODE: 'code' + REGISTRATION_CONTEXT: + EMAIL: 'email' + GENERIC_INVITE: 'generic_invite' + PERSONAL_INVITE: 'personal_invite' + SECTION: + ACCOUNT: 'account' + POSTED: 'posted' + VERIFY: 'verify' + LIMIT: 'limit' + HISTORY: 'history' + TYPE: + CODE: 'code' + EMAIL: 'email' + FORM: 'form' + MODE: 'mode' + NAME: 'name' + PASSWORD: 'password' + PHONE: 'phone' + SECTION: 'section' + TERMS: 'terms' diff --git a/app/script/auth/URLParameter.coffee b/app/script/auth/URLParameter.coffee new file mode 100644 index 00000000000..3eb9c5ec85a --- /dev/null +++ b/app/script/auth/URLParameter.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +z.auth.URLParameter = + CONNECT: 'connect' + ENVIRONMENT: 'env' + EXPIRED: 'expired' + INVITE: 'invite' + LOCALE: 'hl' + LOCALYTICS: 'localytics' diff --git a/app/script/auth/ValidationError.coffee b/app/script/auth/ValidationError.coffee new file mode 100644 index 00000000000..bc905dc2ae8 --- /dev/null +++ b/app/script/auth/ValidationError.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.auth ?= {} + +# Authentication error entity. +class z.auth.ValidationError + # Construct a new Authentication error. + constructor: (types, string_identifier) -> + types = [types] if _.isString types + @types = types + @message = z.localization.Localizer.get_text string_identifier diff --git a/app/script/cache/CacheRepository.coffee b/app/script/cache/CacheRepository.coffee new file mode 100644 index 00000000000..3d45438df6c --- /dev/null +++ b/app/script/cache/CacheRepository.coffee @@ -0,0 +1,56 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cache ?= {} + +### +Cache repository for local storage interactions using amplify. + +@todo We have to come up with a smart solution to handle "amplify.store quota exceeded" + This happened when doing "@cache_repository.set_entity user_et" + +### +class z.cache.CacheRepository + # Construct a new Cache Repository. + constructor: -> + @logger = new z.util.Logger 'z.auth.CacheRepository', z.config.LOGGER.OPTIONS + + ### + Deletes cached data. + + @param keep_conversation_input [Boolean] Should conversation input be kept + @param protected_key_patterns [Array] Keys which should NOT be deleted from the cache + + @return [Array] Keys which have been deleted from the cache + ### + clear_cache: (keep_conversation_input = false, protected_key_patterns = [z.storage.StorageKey.AUTH.SHOW_LOGIN]) -> + protected_key_patterns.push z.storage.StorageKey.CONVERSATION.INPUT if keep_conversation_input + deleted_keys = [] + + $.each amplify.store(), (stored_key) -> + should_be_deleted = true + + for protected_key_pattern in protected_key_patterns + should_be_deleted = false if stored_key.startsWith protected_key_pattern + + if should_be_deleted + z.storage.reset_value stored_key + deleted_keys.push stored_key + + return deleted_keys diff --git a/app/script/calling/CallCenter.coffee b/app/script/calling/CallCenter.coffee new file mode 100644 index 00000000000..d30fb2fc616 --- /dev/null +++ b/app/script/calling/CallCenter.coffee @@ -0,0 +1,267 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} + +SUPPORTED_EVENTS = [ + z.event.Backend.CALL.FLOW_ADD + z.event.Backend.CALL.REMOTE_CANDIDATES_ADD + z.event.Backend.CALL.REMOTE_CANDIDATES_UPDATE + z.event.Backend.CALL.REMOTE_SDP + z.event.Backend.CALL.STATE + z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE + z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE +] + +# User repository for all call interactions with the call service. +class z.calling.CallCenter + ### + Extended check for calling support of browser. + @param conversation_id [String] Conversation ID + @return [Boolean] True if calling is supported + ### + @supports_calling: -> + return z.util.Environment.browser.supports.calling + + ### + Extended check for screen sharing support of browser. + @return [Boolean] True if screen sharing is supported + ### + @supports_screen_sharing: -> + return false if z.util.Environment.frontend.is_production() + return z.util.Environment.browser.supports.screen_sharing + + ### + Construct a new Call Center repository. + + @param call_service [z.calling.CallService] Backend REST API call service implementation + @param conversation_repository [z.conversation.ConversationRepository] Repository for conversation interactions + @param user_repository [z.user.UserRepository] Repository for all user and connection interactions + @param audio_repository [z.audio.AudioRepository] Repository for all audio interactions + ### + constructor: (@call_service, @conversation_repository, @user_repository, @audio_repository) -> + @logger = new z.util.Logger 'z.calling.CallCenter', z.config.LOGGER.OPTIONS + + # Telemetry + @telemetry = new z.telemetry.calling.CallTelemetry() + @flow_status = undefined + @timings = ko.observable() + + # Media Handler + @media_devices_handler = new z.calling.handler.MediaDevicesHandler @ + @media_stream_handler = new z.calling.handler.MediaStreamHandler @ + @media_element_handler = new z.calling.handler.MediaElementHandler @ + + # Call Handler + @state_handler = new z.calling.handler.CallStateHandler @ + @signaling_handler = new z.calling.handler.CallSignalingHandler @ + + #Calls + @calls = @state_handler.calls + @joined_call = @state_handler.joined_call + + @subscribe_to_events() + + # Subscribe to amplify topics. + subscribe_to_events: => + amplify.subscribe z.event.WebApp.CALL.EVENT_FROM_BACKEND, @on_event + amplify.subscribe z.event.WebApp.CONVERSATION.EVENT_FROM_BACKEND, @on_event + amplify.subscribe z.event.WebApp.DEBUG.UPDATE_LAST_CALL_STATUS, @store_flow_status + amplify.subscribe z.util.Logger::LOG_ON_DEBUG, @set_logging + + # Un-subscribe from amplify topics. + un_subscribe: -> + @state_handler.un_subscribe() + @signaling_handler.un_subscribe() + amplify.unsubscribeAll z.event.WebApp.CALL.EVENT_FROM_BACKEND + + + ############################################################################### + # Events + ############################################################################### + + ### + Handle incoming backend events. + @param event [Object] Event payload + ### + on_event: (event) => + return if @state_handler.is_handling_notifications() or event.type not in SUPPORTED_EVENTS + + @logger.log @logger.levels.INFO, + "»» Event: '#{event.type}'", {event_object: event, event_json: JSON.stringify event} + if z.calling.CallCenter.supports_calling() + @_on_event_in_supported_browsers event + else + @_on_event_in_unsupported_browsers event + + ### + Backend calling event handling for browsers supporting calling. + @private + @param event [Object] Event payload + ### + _on_event_in_supported_browsers: (event) -> + @telemetry.trace_event event + switch event.type + when z.event.Backend.CALL.FLOW_ADD + @signaling_handler.on_flow_add_event event + when z.event.Backend.CALL.REMOTE_CANDIDATES_ADD, z.event.Backend.CALL.REMOTE_CANDIDATES_UPDATE + @signaling_handler.on_remote_ice_candidates event + when z.event.Backend.CALL.REMOTE_SDP + @signaling_handler.on_remote_sdp event + when z.event.Backend.CALL.STATE + @state_handler.on_call_state event + + ### + Backend calling event handling for browsers not supporting calling. + @private + @param event [Object] Event payload + ### + _on_event_in_unsupported_browsers: (event) -> + switch event.type + when z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE + @user_repository.get_user_by_id @get_creator_id(event), (creator_et) -> + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.UNSUPPORTED_INCOMING_CALL, { + first_name: creator_et.name() + call_id: event.conversation + } + when z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.UNSUPPORTED_INCOMING_CALL + + + ############################################################################### + # Helper functions + ############################################################################### + + ### + Get a call entity. + @param conversation_id [String] Conversation ID of requested call + @return [z.calling.Call] Call entity for conversation ID + ### + get_call_by_id: (conversation_id) -> + return Promise.resolve() + .then => + if conversation_id + for call_et in @calls() when call_et.id is conversation_id + return call_et + throw new z.calling.CallError 'No call for conversation ID found', z.calling.CallError::TYPE.CALL_NOT_FOUND + throw new z.calling.CallError 'No conversation ID given', z.calling.CallError::TYPE.NO_CONVERSATION_ID + + ### + Helper to identify the creator of a call or choose the first joined one. + @private + @param event [Object] Event payload + ### + get_creator_id: (event) -> + if creator_id = event.creator or event.from + return creator_id + else + return user_id for user_id, device_info of event.participants when device_info.state is z.calling.enum.ParticipantState.JOINED + + + ############################################################################### + # Util functions + ############################################################################### + + ### + Count into the flows of a call. + @param conversation_id [String] Conversation ID + ### + count_flows: (conversation_id) => + @get_call_by_id conversation_id + .then (call_et) => + counting = ({flow: flow_et, sound: "/audio/digits/#{i}.mp3"} for flow_et, i in call_et.get_flows()) + counting.reverse() + + _count_flow = => + act = counting.pop() + return if not act + user_name = act.flow.remote_user.name() + @logger.log @logger.levels.INFO, "Sending audio file '#{act.sound}' to flow '#{act.flow.id}' (#{user_name})" + act.flow.inject_audio_file act.sound, _count_flow + + _count_flow() + .catch (error) => + @logger.log @logger.levels.WARN, "No call for conversation '#{conversation_id}' found to count into flows", error + + + ### + Inject audio into all flows of a call. + @param conversation_id [String] Conversation ID + @param file_path [String] Path to audio file + ### + inject_audio: (conversation_id, file_path) => + @get_call_by_id conversation_id + .then (call_et) -> + flow_et.inject_audio_file file_path for flow_et in call_et.get_flows() + .catch (error) => + @logger.log @logger.levels.WARN, "No call for conversation '#{conversation_id}' found to inject audio into flows", error + + + ############################################################################### + # Logging + ############################################################################### + + # Log call sessions + log_sessions: => + @telemetry.log_sessions() + + print_call_states: => + session_id = 'unknown' + for call_et in @calls() + @logger.force_log "Call state for conversation: #{call_et.id}\n" + session_id = call_et.log_state() + return "session id is : #{session_id}" + + # Report a call for call analysis + report_call: => + send_report = (custom_data) => + Raygun.send new Error('Call failure report'), custom_data + @logger.log @logger.levels.INFO, + "Reported status of flow id '#{custom_data.meta.flow_id}' for call analysis", custom_data + + call_et = @_find_ongoing_call() + if call_et + send_report flow_et.report_status() for flow_et in call_et.get_flows() + else if @flow_status + send_report @flow_status + else + @logger.log @logger.levels.WARN, 'Could not find flows to report for call analysis' + + # Set logging on adapter.js + set_logging: (is_logging_enabled) => + @logger.log @logger.levels.INFO, "Set logging for webRTC Adapter: #{is_logging_enabled}" + adapter?.disableLog = not is_logging_enabled + + # Store last flow status + store_flow_status: (flow_status) => + @flow_status = flow_status if flow_status + + ### + Please solely use this method for logging purposes! It's not intended to do actual work / heavy lifting. + + @private + @param conversation_id [String] Conversation ID + @return [z.calling.Call] Returns an ongoing call entity + ### + _find_ongoing_call: (conversation_id) -> + @get_call_by_id conversation_id + .then (call_et) -> + return call_et + .catch => + return call_et for call in @calls() when call.state() not in z.calling.enum.CallStateGroups.IS_ENDED diff --git a/app/script/calling/CallError.coffee b/app/script/calling/CallError.coffee new file mode 100644 index 00000000000..4505921fe24 --- /dev/null +++ b/app/script/calling/CallError.coffee @@ -0,0 +1,42 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} + +class z.calling.CallError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @type = type + @stack = (new Error()).stack + + @:: = new Error() + @::constructor = @ + @::TYPE = { + CALL_NOT_FOUND: 'z.calling.CallError::TYPE.CALL_NOT_FOUND' + CONVERSATION_EMPTY: 'z.calling.CallError::TYPE.CONVERSATION_EMPTY' + CONVERSATION_TOO_BIG: 'z.calling.CallError::TYPE.CONVERSATION_TOO_BIG' + FLOW_NOT_FOUND: 'z.calling.CallError::TYPE.FLOW_NOT_FOUND' + NO_CAMERA_FOUND: 'z.calling.CallError::TYPE.NO_CAMERA_FOUND' + NO_CONVERSATION_ID: 'z.calling.CallError::TYPE.NO_CONVERSATION_ID' + NO_DEVICES_FOUND: 'z.calling.CallError::TYPE.NO_DEVICES_FOUND' + NO_MICROPHONE_FOUND: 'z.calling.CallError::TYPE.NO_MICROPHONE_FOUND' + NOT_SUPPORTED: 'z.calling.CallError::TYPE.NOT_SUPPORTED' + VOICE_CHANNEL_FULL: 'z.calling.CallError::TYPE.VOICE_CHANNEL_FULL' + } diff --git a/app/script/calling/CallService.coffee b/app/script/calling/CallService.coffee new file mode 100644 index 00000000000..31297038946 --- /dev/null +++ b/app/script/calling/CallService.coffee @@ -0,0 +1,148 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} + +class z.calling.CallService + constructor: (@client) -> + @logger = new z.util.Logger 'z.calling.CallService', z.config.LOGGER.OPTIONS + + ### + Deletes a flow on the backend. + + @note If a call connection with a remote device (participant) ends, we need to delete the flow to this device. + + Sometimes calls to a remote device ended already but the backend state still has active flows to these devices. + In this case we need to force the deletion of these active flows with the reason "released". + + If we detect on the PeerConnection level, that a remote participant dropped, we can + delete the flow to them with reason "timeout". + + Possible errors: + - {"code":404,"message":"the requested flow does not exist","label":"not-found"} + - {"code":403,"message":"cannot remove active flow","label":"in-use"} + + @param delete_flow_info [Object] Info needed to delete a flow + @option delete_flow_info [String] conversation_id + @option delete_flow_info [String] flow_id + @param callbacks [Array] Callbacks after the request has been made + ### + delete_flow: (delete_flow_info, callbacks) -> + reason = delete_flow_info.reason + url = "/conversations/#{delete_flow_info.conversation_id}/call/flows/#{delete_flow_info.flow_id}" + url += "?reason=#{reason}" if reason + + @client.send_request + type: 'DELETE' + api_endpoint: '/conversations/{conversation_id}/call/flows/{flow_id}' + url: @client.create_url url + callback: callbacks + + ### + Lists existing call flows for a specific conversation. + + @param conversation_id [String] Conversation ID + @param callbacks [Array] Callbacks after the request has been made + ### + get_flows: (conversation_id, callbacks) -> + @client.send_request + type: 'GET' + api_endpoint: '/conversations/{conversation_id}/call/flows' + url: @client.create_url "/conversations/#{conversation_id}/call/flows" + callback: callbacks + + ### + Returns the participants and their call states in a specified conversation. + + @param conversation_id [String] Conversation ID + @param callbacks [Array] Callbacks after the request has been made + ### + get_state: (conversation_id, callbacks) -> + @client.send_request + type: 'GET' + api_endpoint: '/conversations/{conversation_id}/call/state' + url: @client.create_url "/conversations/#{conversation_id}/call/state" + callback: callbacks + + ### + Commands the backend to create a flow. + + @param conversation_id [String] Conversation ID + @param callbacks [Array] Callbacks after the request has been made + ### + post_flows: (conversation_id, callbacks) -> + @client.send_request + type: 'POST' + api_endpoint: '/conversations/{conversation_id}/call/flows' + url: @client.create_url "/conversations/#{conversation_id}/call/flows" + callback: callbacks + + ### + Add an ICE candidate. + + @param conversation_id [String] Conversation ID + @param flow_id [String] Flow ID + @param ice_info [z.calling.payloads.ICECandidateInfo] Signaling info bundled with ICE candidate + @param callbacks [Array] Callbacks after the request has been made + ### + post_local_candidates: (conversation_id, flow_id, ice_info, callbacks) -> + @client.send_json + type: 'POST' + api_endpoint: '/conversations/{conversation_id}/call/flows/{flow_id}/local_candidates' + url: @client.create_url "/conversations/#{conversation_id}/call/flows/#{flow_id}/local_candidates" + data: + candidates: [ice_info] + callback: callbacks + + ### + Update the SDP of a connection. + + @note Errors can be: + - {"code":400,"message":"invalid SDP transition requested","label":"bad-sdp"} + + The "bad-sdp" can happen when you send an offer to an offer or if one flow has been already partially negotiated and we try to negotiate for a second flow. + + @param conversation_id [String] Conversation ID + @param flow_id [String] Flow ID + @param sdp [z.calling.SDPInfo] Signaling info bundled with SDP + @param callbacks [Array] Callbacks after the request has been made + ### + put_local_sdp: (conversation_id, flow_id, sdp, callbacks) -> + @client.send_json + type: 'PUT' + api_endpoint: '/conversations/{conversation_id}/call/flows/{flow_id}/local_sdp' + url: @client.create_url "/conversations/#{conversation_id}/call/flows/#{flow_id}/local_sdp" + data: sdp + callback: callbacks + + ### + Returns the current state of the client and all participants. + + @param conversation_id [String] Conversation ID + @param payload [Object] Participant payload to be set + @param callbacks [Array] Callbacks after the request has been made + ### + put_state: (conversation_id, payload, callbacks) -> + @client.send_json + type: 'PUT' + api_endpoint: '/conversations/{conversation_id}/call/state' + url: @client.create_url "/conversations/#{conversation_id}/call/state" + data: + self: payload + callback: callbacks diff --git a/app/script/calling/CallTrackingInfo.coffee b/app/script/calling/CallTrackingInfo.coffee new file mode 100644 index 00000000000..f3ff8f7c999 --- /dev/null +++ b/app/script/calling/CallTrackingInfo.coffee @@ -0,0 +1,34 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} + +class z.calling.CallTrackingInfo + constructor: (params) -> + @conversation_id = params.conversation_id + @session_id = params.session_id + @time_started = new Date() + @participants_joined = {} + + add_participant: (participant_et) -> + @participants_joined[participant_et.user.name()] = true + + to_string: -> + participants = Object.keys(@participants_joined).join ', ' + return "#{@session_id} in #{@conversation_id} | #{@time_started.toUTCString()} | To: #{participants}" diff --git a/app/script/calling/entities/Call.coffee b/app/script/calling/entities/Call.coffee new file mode 100644 index 00000000000..cef8c71565d --- /dev/null +++ b/app/script/calling/entities/Call.coffee @@ -0,0 +1,518 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.entities ?= {} + +# Call entity. +class z.calling.entities.Call + ### + Construct a new call entity. + @param conversation_et [z.entity.Conversation] Conversation the call takes place in + @param self_user [z.entity.User] Self user entity + ### + constructor: (@conversation_et, @self_user, @telemetry) -> + @logger = new z.util.Logger "z.calling.Call (#{@conversation_et.id})", z.config.LOGGER.OPTIONS + + # IDs and references + @id = @conversation_et.id + @session_id = ko.observable undefined + @event_sequence = 0 + + # States + @call_timer_interval = undefined + @timer_start = undefined + @duration_time = ko.observable 0 + @finished_reason = z.calling.enum.CallFinishedReason.UNKNOWN + @remote_media_type = ko.observable z.calling.enum.MediaType.NONE + + @is_connected = ko.observable false + @is_group = @conversation_et.is_group + @is_remote_screen_shared = ko.pureComputed => + return @remote_media_type() is z.calling.enum.MediaType.SCREEN + @is_remote_videod = ko.pureComputed => + return @remote_media_type() in [z.calling.enum.MediaType.SCREEN, z.calling.enum.MediaType.VIDEO] + + @self_client_joined = ko.observable false + @self_user_joined = ko.observable false + @state = ko.observable z.calling.enum.CallState.UNKNOWN + @previous_state = undefined + @is_declined_timer = undefined + + # user declined group call + @is_declined = ko.observable false + @is_declined.subscribe (is_declined) => + @_stop_call_sound true if is_declined + + @self_user_joined.subscribe (is_joined) => + @is_declined false if is_joined + + # User entities + @creator = ko.observable undefined + + # @todo Calculate panning on participants update + @participants = ko.observableArray [] + @participants_count = ko.pureComputed => + if @self_user_joined() + @get_number_of_participants() + 1 + else + @get_number_of_participants() + @max_number_of_participants = 0 + + @interrupted_participants = ko.observableArray [] + + # Media + @local_audio_stream = ko.observable() + @local_video_stream = ko.observable() + + # Statistics + @_reset_timer() + + # Observable subscriptions + @is_connected.subscribe (is_connected) => + if is_connected + @telemetry.track_event z.tracking.EventName.CALLING.ESTABLISHED_CALL, @ + @timer_start = Date.now() - 100 + @call_timer_interval = window.setInterval => + @update_timer_duration() + , 1000 + + @participants_count.subscribe (users_in_call) => + @max_number_of_participants = Math.max users_in_call, @max_number_of_participants + + @self_client_joined.subscribe (is_joined) => + if is_joined + if @state() isnt z.calling.enum.CallState.OUTGOING + amplify.publish z.event.WebApp.CALL.SIGNALING.POST_FLOWS, @id + else + @is_connected false + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.CALL_DROP + @telemetry.track_duration @ + @_reset_timer() + @_reset_flows() + + @is_ongoing_on_another_client = ko.pureComputed => + return @self_user_joined() and not @self_client_joined() + + @network_interruption = ko.pureComputed => + if @is_connected() + if @is_group() + return @interrupted_participants().length > 0 and @interrupted_participants().length is @participants().length + else + return @interrupted_participants().length > 0 + return false + + @network_interruption.subscribe (is_interrupted) -> + if is_interrupted + amplify.publish z.event.WebApp.AUDIO.PLAY_IN_LOOP, z.audio.AudioType.NETWORK_INTERRUPTION + else + amplify.publish z.event.WebApp.AUDIO.STOP, z.audio.AudioType.NETWORK_INTERRUPTION + + @state.subscribe (state) => + @logger.log @logger.levels.DEBUG, "Call state changed to '#{state}'" + + @_clear_join_timer() if @is_group() + + switch state + when z.calling.enum.CallState.CONNECTING + @_on_state_connecting() + when z.calling.enum.CallState.INCOMING + @_on_state_incoming() + when z.calling.enum.CallState.DELETED + @_on_state_deleted() + when z.calling.enum.CallState.IGNORED + @_on_state_ignored() + when z.calling.enum.CallState.ONGOING + @_on_state_ongoing() + when z.calling.enum.CallState.OUTGOING + @_on_state_outgoing() + + @previous_state = state + + update_timer_duration: => + @duration_time Math.floor (Date.now() - @timer_start) / 1000 + + + ############################################################################### + # Call states + ############################################################################### + + _on_state_connecting: => + @_stop_call_sound @previous_state is z.calling.enum.CallState.INCOMING + @is_declined false + + _on_state_incoming: => + @_play_call_sound true + @_group_call_timeout true if @is_group() + + _on_state_deleted: => + if @previous_state in z.calling.enum.CallStateGroups.IS_RINGING + @_stop_call_sound @previous_state is z.calling.enum.CallState.INCOMING + @is_declined false + + _on_state_ignored: => + if @previous_state in z.calling.enum.CallStateGroups.IS_RINGING + @_stop_call_sound @previous_state is z.calling.enum.CallState.INCOMING + + _on_state_ongoing: => + if @previous_state in z.calling.enum.CallStateGroups.IS_RINGING + @_stop_call_sound @previous_state is z.calling.enum.CallState.INCOMING + + _on_state_outgoing: => + @_play_call_sound false + @_group_call_timeout false if @is_group() + + _clear_join_timer: => + window.clearTimeout @is_declined_timer + @is_declined_timer = undefined + + _group_call_timeout: (is_incoming) => + @is_declined_timer = window.setTimeout => + @_stop_call_sound is_incoming + @is_declined true if is_incoming + , 30000 + + _play_call_sound: (is_incoming) -> + if is_incoming + amplify.publish z.event.WebApp.AUDIO.PLAY_IN_LOOP, z.audio.AudioType.INCOMING_CALL + else + amplify.publish z.event.WebApp.AUDIO.PLAY_IN_LOOP, z.audio.AudioType.OUTGOING_CALL + + _stop_call_sound: (is_incoming) -> + if is_incoming + amplify.publish z.event.WebApp.AUDIO.STOP, z.audio.AudioType.INCOMING_CALL + else + amplify.publish z.event.WebApp.AUDIO.STOP, z.audio.AudioType.OUTGOING_CALL + + + ############################################################################### + # State handling + ############################################################################### + + # Check whether this call is a video call + update_remote_state: (participants) -> + media_type_updated = false + for participant_id, state of participants when participant_id isnt @self_user.id + participant_et = @get_participant_by_id participant_id + if participant_et + participant_et.state.muted state.muted if state.muted? + participant_et.state.screen_shared state.screen_shared if state.screen_shared? + participant_et.state.videod state.videod if state.videod? + + if state.screen_shared + @remote_media_type z.calling.enum.MediaType.SCREEN + media_type_updated = true + else if state.videod + @remote_media_type z.calling.enum.MediaType.VIDEO + media_type_updated = true + + @remote_media_type z.calling.enum.MediaType.AUDIO if not media_type_updated + + # Ignore a call. + ignore: => + if @is_group() + @is_declined true # TODO would be nice if group call could have also an ignored state + else + @state z.calling.enum.CallState.IGNORED + + + ############################################################################### + # Participants + ############################################################################### + + ### + Add a participant to the call. + @param participant_et [z.calling.Participant] Participant entity to be added to the call + @return [Boolean] Has the participant been added + ### + add_participant: (participant_et) => + if not @get_participant_by_id participant_et.user.id + @participants.push participant_et + @logger.log @logger.levels.DEBUG, "Participants updated: '#{participant_et.user.name()}' added'" + return true + return false + + ### + Remove a participant from the call. + + @param participant_et [z.calling.Participant] Participant entity to be removed from the call + @param delete_on_backend [Boolean] Should the flow with the participant be removed on the backend + @return [Boolean] Has the participant been removed + ### + delete_participant: (participant_et, delete_on_backend = true) => + return false if not @get_participant_by_id participant_et.user.id + + # Delete participant + @participants.remove participant_et + # Delete flows from participant (if any left) + flow_et = participant_et.get_flow() + if flow_et + @_delete_flow_by_id flow_et, delete_on_backend + + @logger.log @logger.levels.DEBUG, "Participants updated: '#{participant_et.user.name()}' removed" + if @get_number_of_participants() is 0 + amplify.publish z.event.WebApp.CALL.STATE.DELETE, @id + return true + + ### + Get the number of participants in the call. + @return [Number] Number of participants in call excluding the self user + ### + get_number_of_participants: => + return @participants().length + + ### + Get a call participant by his id. + @param user_id [String] User ID of participant to be returned + @return [z.calling.Participant] Participant that matches given user ID + ### + get_participant_by_id: (user_id) => + return participant for participant in @participants() when participant.user.id is user_id + + ### + Set a user as the creator of the call. + @param user_et [z.entity.User] User entity to be set as call creator + ### + set_creator: (user_et) => + if not @creator() + @logger.log @logger.levels.INFO, "Call created by: #{user_et.name()}" + @creator user_et + + ### + Update the remote participants of a call. + + @param participants_ets [Array] Array joined call participants + @param sequential_event [Boolean] Should the update be limited to one change only + + @note Some backend 'call-state' events contain false information + If a call event is sequential to the previous one (meaning the sequence number is increased by one) and the + 'event.cause' of the call state event is 'requested' (as it was triggered by another client PUTting its state) + then the delta in participants can only be one. If we have added a user, we cannot add or remove another one. + ### + update_participants: (participant_ets = [], sequential_event = false) => + sequential_event = false if @state() in z.calling.enum.CallStateGroups.IS_RINGING + if sequential_event + @logger.log @logger.levels.INFO, 'Sequential event by request: Only one participant change will be applied' + + # Add active participants + for participant_et in participant_ets when @add_participant participant_et + participant_joined = true + break if sequential_event + + # Find inactive participants + if participant_ets.length isnt @get_number_of_participants() and (not sequential_event or not participant_joined) + active_participant_ids = (participant_et.user.id for participant_et in participant_ets) + delete_participants_ets = ( + participant_et for participant_et in @participants() when participant_et.user.id not in active_participant_ids + ) + + # Remove inactive participants + if delete_participants_ets?.length > 0 + for participant_et in delete_participants_ets when @delete_participant participant_et + participant_left = true + break if sequential_event + if participant_left and @self_client_joined() + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.CALL_DROP + + @_sort_participants_by_panning() + + + ############################################################################### + # Flows + ############################################################################### + + ### + Construct and add a flow to a call participant. + + @param flow_id [String] ID of flow to be constructed + @param user_et [z.entity.User] User that the flow is with + @param audio_context [AudioContext] Audio context for the flow audio + @param call_timings [z.telemetry.calling.CallSetupTimings] Optional object to track duration of call setup + ### + construct_flow: (flow_id, user_et, audio_context, call_timings) => + participant_et = @get_participant_by_id user_et.id + + create_flow = (flow_id, participant_et) => + if call_timings + flow_timings = $.extend new z.telemetry.calling.CallSetupTimings(@id), call_timings.get() + flow_timings.time_step z.telemetry.calling.CallSetupSteps.FLOW_RECEIVED + flow_timings.flow_id = flow_id + flow_et = new z.calling.entities.Flow flow_id, @, participant_et, audio_context, flow_timings + participant_et.add_flow flow_et + return flow_et + + if participant_et + # We have to update the user info + if @get_flow_by_id flow_id + @logger.log @logger.levels.WARN, "Not adding flow '#{flow_id}' as it already exists" + else + @logger.log @logger.levels.DEBUG, "Adding flow '#{flow_id}' to participant '#{participant_et.user.name()}'" + create_flow flow_id, participant_et + else + participant_et = new z.calling.entities.Participant user_et + @add_participant participant_et + @logger.log @logger.levels.DEBUG, "Adding flow '#{flow_id}' to new participant '#{participant_et.user.name()}'" + create_flow flow_id, participant_et + + ### + Get the flow that matches the given ID. + @param flow_id [String] ID of flow to be returned + @return [z.calling.Flow] Matching flow entity + ### + get_flow_by_id: (flow_id) => + return flow_et for flow_et in @get_flows() when flow_et.id is flow_id + + ### + Get all flows of the call. + @return [Array] Array of flows + ### + get_flows: => + return (participant_et.get_flow() for participant_et in @participants() when participant_et.get_flow()) + + ### + Get the flow to a specific user. + @param user_id [String] User ID that the flow connects to + @return [z.calling.Flow] Flow to given user + ### + get_flows_by_user_id: (user_id) => + return @get_participant_by_id(user_id).flows() + + ### + Get full flow telemetry report of the call. + @return [Array] Array of flows + ### + get_flow_telemetry: => + return (participant.get_flow()?.get_telemetry() for participant in @participants() when participant.get_flow()) + + ### + Get the number of flows of the call. + @return [Number] Number of flows + ### + get_number_of_flows: => + return @get_flows().length + + ### + Get the number of active flows of the call. + @return [Number] Number of active flows + ### + get_number_of_active_flows: => + return (flow_et for flow_et in @get_flows() when flow_et.is_active).length or 0 + + ### + Delete a flow with a given ID. + + @private + @param flow_et [z.calling.Flow] Flow entity to be deleted + @param delete_on_backend [Boolean] Should the flow with the participant be removed on the backend + ### + _delete_flow_by_id: (flow_et, delete_on_backend = true) => + return if not flow_et + flow_et.reset_flow() + + return if not delete_on_backend + # Delete flow on backend + flow_deletion_info = new z.calling.payloads.FlowDeletionInfo @id, flow_et.id + amplify.publish z.event.WebApp.CALL.SIGNALING.DELETE_FLOW, flow_deletion_info + + ############################################################################### + # Panning + ############################################################################### + + ### + Calculates the panning (from left to right) to position a user in a group call. + + @note The deal is to calculate Jenkins' one-at-a-time hash (JOAAT) for each participant and then + sort all participants in an array by their JOAAT hash. After that the array index of each user + is used to allocate the position with the return value of this function. + + @param index [Number] Index of a user in a sorted array + @param total [Number] Number of users + @return [Number] Panning in the range of -1 to 1 with -1 on the left + ### + _calculate_panning: (index, total) -> + return 0.0 if total is 1 + + pos = -(total - 1.0) / (total + 1.0) + delta = (-2.0 * pos) / (total - 1.0) + + return pos + delta * index + + # Sort the call participants by their audio panning. + _sort_participants_by_panning: -> + return if @participants().length < 2 + + # Sort by JOOAT Hash and calculate panning + @participants.sort (participant_a, participant_b) -> + return participant_a.user.joaat_hash - participant_b.user.joaat_hash + + for participant_et, i in @participants() + panning = @_calculate_panning i, @participants().length + @logger.log @logger.levels.INFO, + "Panning for '#{participant_et.user.name()}' recalculated to '#{panning}'" + participant_et.panning panning + + panning_info = (participant_et.user.name() for participant_et in @participants()).join ', ' + @logger.log @logger.levels.INFO, "New panning order: #{panning_info}" + + + ############################################################################### + # Reset + ############################################################################### + + ### + Reset the call states. + @private + ### + reset_call: => + @self_client_joined false + @event_sequence = 0 + @finished_reason = z.calling.enum.CallFinishedReason.UNKNOWN + @is_connected false + @session_id undefined + @self_user_joined false + @is_declined false + + ### + Reset the call timers. + @private + ### + _reset_timer: -> + window.clearInterval @call_timer_interval if @call_timer_interval + @timer_start = undefined + @duration_time 0 + + ### + Reset all flows of the call. + @private + ### + _reset_flows: -> + @_delete_flow_by_id flow_et for flow_et in @get_flows() + + + ############################################################################### + # Logging + ############################################################################### + + # Log flow status to console. + log_status: => + flow_et.log_status() for flow_et in @get_flows() + + # Log flow setup step timings to console. + log_timings: => + flow_et.log_timings() for flow_et in @get_flows() diff --git a/app/script/calling/entities/Flow.coffee b/app/script/calling/entities/Flow.coffee new file mode 100644 index 00000000000..f68dd9cfee0 --- /dev/null +++ b/app/script/calling/entities/Flow.coffee @@ -0,0 +1,946 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.entities ?= {} + +# Static array of where to put people in the stereo scape. +AUDIO_BITRATE = '30' +AUDIO_PTIME = '60' + +# Flow entity. +class z.calling.entities.Flow + ### + Construct a new flow entity. + + @param id [String] ID of the flow + @param call_et [z.calling.Call] Call entity that the flow belongs to + @param participant_et [z.calling.Participant] Participant entity that the flow belongs to + @param audio_context [AudioContext] AudioContext to be used with the flow + @param timings [z.telemetry.calling.CallSetupTimings] Timing statistics of call setup steps + ### + constructor: (@id, @call_et, @participant_et, @audio_context, timings) -> + @logger = new z.util.Logger "z.calling.Flow (#{@id})", z.config.LOGGER.OPTIONS + + @conversation_id = @call_et.id + + # States + @converted_own_sdp_state = ko.observable false + @is_active = ko.observable false + @is_answer = ko.observable undefined + @is_group = @call_et.is_group + + # Audio + @audio = new z.calling.entities.FlowAudio @, @audio_context + + # ICE candidates + @ice_candidates_cache = [] + + # Users + @creator_user_id = ko.observable undefined + @remote_user = @participant_et.user + @remote_user_id = @remote_user.id + @self_user_id = @call_et.self_user.id + + # Telemetry + @telemetry = new z.telemetry.calling.FlowTelemetry @id, @remote_user_id, @call_et, timings + + @is_answer.subscribe (is_answer) => @telemetry.update_is_answer is_answer + + @creator_user_id.subscribe (user_id) => + if user_id is @self_user_id + @logger.log @logger.levels.INFO, "Creator: We are the official '#{z.calling.rtc.SDPType.OFFER}'" + @is_answer false + else + @logger.log @logger.levels.INFO, "Creator: We are the official '#{z.calling.rtc.SDPType.ANSWER}'" + @is_answer true + + + ############################################################################### + # PeerConnection + ############################################################################### + + @peer_connection = undefined + @payload = ko.observable undefined + @pc_initialized = ko.observable false + @pc_initialized.subscribe (is_initialized) => + @telemetry.set_peer_connection @peer_connection + @telemetry.schedule_check 5000 if is_initialized + + @audio_stream = @call_et.local_audio_stream + @video_stream = @call_et.local_video_stream + + @has_media_stream = ko.pureComputed => return @video_stream()? or @audio_stream()? + + @connection_state = ko.observable z.calling.rtc.ICEConnectionState.NEW + @gathering_state = ko.observable z.calling.rtc.ICEGatheringState.NEW + @signaling_state = ko.observable z.calling.rtc.SignalingState.NEW + + @connection_state.subscribe (ice_connection_state) => + switch ice_connection_state + when z.calling.rtc.ICEConnectionState.CHECKING + @telemetry.time_step z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CHECKING + + when z.calling.rtc.ICEConnectionState.COMPLETED, z.calling.rtc.ICEConnectionState.CONNECTED + @telemetry.start_statistics ice_connection_state + @call_et.is_connected true + @participant_et.is_connected true + @call_et.interrupted_participants.remove @participant_et + @call_et.state z.calling.enum.CallState.ONGOING + + when z.calling.rtc.ICEConnectionState.DISCONNECTED + @participant_et.is_connected false + @call_et.interrupted_participants.push @participant_et + @is_answer false + @negotiation_mode z.calling.enum.SDPNegotiationMode.ICE_RESTART + + when z.calling.rtc.ICEConnectionState.FAILED + @participant_et.is_connected false + if @is_group() + @call_et.interrupted_participants.remove @participant_et + @call_et.delete_participant @participant_et if @call_et.self_client_joined() + else + amplify.publish z.event.WebApp.CALL.STATE.LEAVE, @call_et.id + + when z.calling.rtc.ICEConnectionState.CLOSED + @participant_et.is_connected false + @call_et.delete_participant @participant_et if @call_et.self_client_joined() + + @signaling_state.subscribe (signaling_state) => + if signaling_state is z.calling.rtc.SignalingState.CLOSED and not @converted_own_sdp_state() + @logger.log @logger.levels.DEBUG, "PeerConnection with '#{@remote_user.name()}' was closed" + @call_et.delete_participant @participant_et + @_remove_media_streams() + if not @is_group() + @call_et.finished_reason = z.calling.enum.CallFinishedReason.CONNECTION_DROPPED + + @negotiation_mode = ko.observable z.calling.enum.SDPNegotiationMode.DEFAULT + @negotiation_needed = ko.observable false + + + ############################################################################### + # Local SDP + ############################################################################### + + @local_sdp_type = ko.observable undefined + @local_sdp = ko.observable undefined + @local_sdp.subscribe (sdp) => + if sdp + @local_sdp_type sdp.type + if @has_sent_local_sdp() + @has_sent_local_sdp false + @should_add_local_sdp true + + @has_sent_local_sdp = ko.observable false + @should_add_local_sdp = ko.observable true + + @send_sdp_timeout = undefined + + @can_set_local_sdp = ko.pureComputed => + in_connection_progress = @connection_state() is z.calling.rtc.ICEConnectionState.CHECKING + progress_gathering_states = [z.calling.rtc.ICEGatheringState.GATHERING, z.calling.rtc.ICEGatheringState.COMPLETE] + in_progress = in_connection_progress and @gathering_state() in progress_gathering_states + + in_offer_state = @local_sdp_type() is z.calling.rtc.SDPType.OFFER + in_wrong_state = in_offer_state and @signaling_state() is z.calling.rtc.SignalingState.REMOTE_OFFER + is_blocked = @signaling_state() is z.calling.rtc.SignalingState.CLOSED + + return @local_sdp() and @should_add_local_sdp() and not is_blocked and not in_progress and not in_wrong_state + + @can_set_local_sdp.subscribe (can_set) => + if can_set + @logger.log @logger.levels.DEBUG, "State changed - can_set_local_sdp: #{can_set}" + @_set_local_sdp() + + + ############################################################################### + # Remote SDP + ############################################################################### + + @remote_sdp_type = ko.observable undefined + @remote_sdp = ko.observable undefined + @remote_sdp.subscribe (sdp) => + if sdp + @remote_sdp_type sdp.type + @should_add_remote_sdp true + + @should_add_remote_sdp = ko.observable false + + @can_set_remote_sdp = ko.pureComputed => + is_remote_offer = @remote_sdp_type() is z.calling.rtc.SDPType.OFFER + have_local_offer = @signaling_state() is z.calling.rtc.SignalingState.LOCAL_OFFER + in_wrong_state = is_remote_offer and have_local_offer + + return @pc_initialized() and @should_add_remote_sdp() and not in_wrong_state + + @can_set_remote_sdp.subscribe (can_set) => + if can_set + @logger.log @logger.levels.DEBUG, "State changed - can_set_remote_sdp: '#{can_set}'" + @_set_remote_sdp() + .then => + if @has_sent_local_sdp() and @remote_sdp().type is z.calling.rtc.SDPType.OFFER + @is_answer true + + + ############################################################################### + # Gates + ############################################################################### + + @can_create_sdp = ko.pureComputed => + in_state_for_creation = @negotiation_needed() and @signaling_state() isnt z.calling.rtc.SignalingState.CLOSED + can_create = @pc_initialized() and in_state_for_creation + @logger.log @logger.levels.OFF, "State recalculated - can_create_answer: #{can_create}" + return can_create + + @can_create_sdp.subscribe (can_create) => + if can_create + @logger.log @logger.levels.DEBUG, "State changed - can_create_sdp: #{can_create}" + + @can_create_answer = ko.pureComputed => + answer_state = @is_answer() and @signaling_state() is z.calling.rtc.SignalingState.REMOTE_OFFER + can_create = @can_create_sdp() and answer_state + @logger.log @logger.levels.OFF, "State recalculated - can_create_answer: #{can_create}" + return can_create + + @can_create_answer.subscribe (can_create) => + if can_create + @logger.log @logger.levels.DEBUG, "State changed - can_create_answer: #{can_create}" + @negotiation_needed false + @_create_answer() + + @can_create_offer = ko.pureComputed => + offer_state = not @is_answer() and @signaling_state() is z.calling.rtc.SignalingState.STABLE + can_create = @can_create_sdp() and offer_state + @logger.log @logger.levels.OFF, "State recalculated - can_create_offer: #{can_create}" + return can_create + + @can_create_offer.subscribe (can_create) => + if can_create + @logger.log @logger.levels.DEBUG, "State changed - can_create_offer: #{can_create}" + @negotiation_needed false + @_create_offer() + + @can_initialize_peer_connection = ko.pureComputed => + can_initialize = @has_media_stream() and @payload() and not @pc_initialized() + @logger.log @logger.levels.OFF, "State recalculated - can_initialize_peer_connection: #{can_initialize}" + return can_initialize + + @can_initialize_peer_connection.subscribe (can_initialize) => + if can_initialize + @logger.log @logger.levels.DEBUG, "State changed - can_initialize_peer_connection: #{can_initialize}" + @_initialize_peer_connection() + + @can_set_ice_candidates = ko.pureComputed => + can_set = @local_sdp() and @remote_sdp() and @signaling_state() is z.calling.rtc.SignalingState.STABLE + @logger.log @logger.levels.OFF, "State recalculated - can_set_ice_candidates: #{can_set}" + return can_set + + @can_set_ice_candidates.subscribe (can_set) => + if can_set + @logger.log @logger.levels.DEBUG, "State changed - can_set_ice_candidates: #{can_set}" + @_add_cached_ice_candidates() + + @logger.log @logger.levels.INFO, "Flow has an initial panning of '#{@participant_et.panning()}'" + + + ############################################################################### + # Payload handling + ############################################################################### + + ### + Add the payload to the flow. + @note Magic here is that if the remote_user is not the creator then the creator *MUST* be us even if creator is null + @param payload [RTCConfiguration] Configuration to be used to set up the PeerConnection + ### + add_payload: (payload) => + @logger.log @logger.levels.INFO, "Setting payload to be used for flow with '#{@remote_user.name()}'" + return @logger.log @logger.levels.WARN, 'Payload already set' if @payload() + + @creator_user_id payload.creator + @payload @_rewrite_payload payload + if payload.remote_user isnt payload.creator + @logger.log @logger.levels.INFO, "We are the creator of flow with user '#{@remote_user.name()}'" + @is_answer false + else + @logger.log @logger.levels.INFO, "We are not the creator of flow with user '#{@remote_user.name()}'" + @is_answer true + + @is_active payload.active + @audio.hookup payload.active + + ### + Rewrite the payload to be standards compliant. + + @private + @param payload [RTCConfiguration] Payload to be rewritten + @return [RTCConfiguration] Rewritten payload + ### + _rewrite_payload: (payload) -> + for ice_server in payload.ice_servers when not ice_server.urls + ice_server.urls = [ice_server.url] + return payload + + + ############################################################################### + # PeerConnection handling + ############################################################################### + + ### + Close the PeerConnection. + @private + ### + _close_peer_connection: -> + @logger.log @logger.levels.INFO, "Closing PeerConnection with '#{@remote_user.name()}'" + if @peer_connection? + @peer_connection.onsignalingstatechange = => + @logger.log @logger.levels.DEBUG, "State change ignored - signaling state: #{@peer_connection.signalingState}" + @peer_connection.close() + @logger.log @logger.levels.DEBUG, 'Closing PeerConnection successful' + + ### + Create the PeerConnection configuration. + @private + @return [RTCConfiguration] Configuration object to initialize PeerConnection + ### + _configure_peer_connection: -> + return {} = + iceServers: @payload().ice_servers + bundlePolicy: 'max-bundle' + rtcpMuxPolicy: 'require' + + ### + Initialize the PeerConnection for the flow. + @see https://developer.mozilla.org/en-US/docs/Web/API/RTCConfiguration + @private + ### + _create_peer_connection: -> + @peer_connection = new window.RTCPeerConnection @_configure_peer_connection() + @telemetry.time_step z.telemetry.calling.CallSetupSteps.PEER_CONNECTION_CREATED + @signaling_state @peer_connection.signalingState + @logger.log @logger.levels.DEBUG, "PeerConnection with '#{@remote_user.name()}' created", @payload().ice_servers + + ### + A MediaStream was added to the PeerConnection. + @param event [MediaStreamEvent] Event that contains the newly added MediaStream + ### + @peer_connection.onaddstream = (event) => + @logger.log @logger.levels.DEBUG, 'Remote MediaStream added to PeerConnection', + {stream: event.stream, audio_tracks: event.stream.getAudioTracks(), video_tracks: event.stream.getVideoTracks()} + media_stream = z.calling.handler.MediaStreamHandler.detect_media_stream_type event.stream + if media_stream.type is z.calling.enum.MediaType.AUDIO + media_stream = @audio.wrap_speaker_stream event.stream + media_stream_info = new z.calling.payloads.MediaStreamInfo z.calling.enum.MediaStreamSource.REMOTE, @id, media_stream, @call_et + amplify.publish z.event.WebApp.CALL.MEDIA.ADD_STREAM, media_stream_info + + ### + A MediaStreamTrack was added to the PeerConnection. + @param event [MediaStreamTrackEvent] Event that contains the newly added MediaStreamTrack + ### + @peer_connection.onaddtrack = (event) => + @logger.log @logger.levels.DEBUG, 'Remote MediaStreamTrack added to PeerConnection', event + + ### + A MediaStream was removed from the PeerConnection. + @param event [MediaStreamEvent] Event that a MediaStream has been removed + ### + @peer_connection.onremovestream = (event) => + @logger.log @logger.levels.DEBUG, 'Remote MediaStream removed from PeerConnection', event + + ### + A MediaStreamTrack was removed from the PeerConnection. + @param event [MediaStreamTrackEvent] Event that a MediaStreamTrack has been removed + ### + @peer_connection.onremovetrack = (event) => + @logger.log @logger.levels.DEBUG, 'Remote MediaStreamTrack removed from PeerConnection', event + + ### + A local ICE candidates is available. + @param event [RTCPeerConnectionIceEvent] Event that contains the generated ICE candidate + ### + @peer_connection.onicecandidate = (event) => + @logger.log @logger.levels.INFO, 'New ICE candidate generated', event + @telemetry.time_step z.telemetry.calling.CallSetupSteps.ICE_GATHERING_STARTED + if @has_sent_local_sdp() + if event.candidate + @_send_ice_candidate event.candidate + else + @logger.log @logger.levels.INFO, 'End of ICE candidates - trickling end candidate' + @_send_ice_candidate @_fake_ice_candidate 'a=end-of-candidates' + else if not event.candidate + @logger.log @logger.levels.INFO, 'End of ICE candidates - sending SDP' + @telemetry.time_step z.telemetry.calling.CallSetupSteps.ICE_GATHERING_COMPLETED + @_send_local_sdp() + + # ICE connection state has changed. + @peer_connection.oniceconnectionstatechange = (event) => + @logger.log @logger.levels.DEBUG, 'State changed - ICE connection', event + return if not @peer_connection or @call_et.state() is z.calling.enum.CallState.DELETED + + @logger.log @logger.levels.LEVEL_1, "ICE connection state: #{@peer_connection.iceConnectionState}" + @logger.log @logger.levels.LEVEL_1, "ICE gathering state: #{@peer_connection.iceGatheringState}" + + @gathering_state @peer_connection.iceGatheringState + @connection_state @peer_connection.iceConnectionState + + # SDP negotiation needed. + @peer_connection.onnegotiationneeded = (event) => + if not @negotiation_needed() + @logger.log @logger.levels.DEBUG, 'State changed - negotiation needed: true', event + @negotiation_needed true + + # Signaling state has changed. + @peer_connection.onsignalingstatechange = (event) => + @logger.log @logger.levels.DEBUG, "State changed - signaling state: #{@peer_connection.signalingState}", event + @signaling_state @peer_connection.signalingState + + # Initialize the PeerConnection. + _initialize_peer_connection: -> + @_create_peer_connection() + @_add_media_streams() + @pc_initialized true + + + ############################################################################### + # SDP handling + ############################################################################### + + ### + Save the remote SDP received from backend within the flow. + @param remote_sdp [RTCSessionDescription] Remote Session Description Protocol + ### + save_remote_sdp: (remote_sdp) => + @logger.log @logger.levels.DEBUG, "Saving remote SDP of type '#{remote_sdp.type}'" + @telemetry.time_step z.telemetry.calling.CallSetupSteps.REMOTE_SDP_RECEIVED + @remote_sdp @_rewrite_sdp remote_sdp, z.calling.enum.SDPSource.REMOTE + + ### + Create a local SDP of type 'answer'. + @see https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/createAnswer + @private + ### + _create_answer: -> + @logger.log @logger.levels.INFO, "Creating '#{z.calling.rtc.SDPType.ANSWER}' for flow with '#{@remote_user.name()}'" + @peer_connection.createAnswer() + .then (sdp_answer) => + @logger.log @logger.levels.DEBUG, "Creating '#{z.calling.rtc.SDPType.ANSWER}' successful", sdp_answer + @telemetry.time_step z.telemetry.calling.CallSetupSteps.LOCAL_SDP_CREATED + @local_sdp @_rewrite_sdp sdp_answer, z.calling.enum.SDPSource.LOCAL + .catch (error) => + @logger.log @logger.levels.ERROR, "Creating '#{z.calling.rtc.SDPType.ANSWER}' failed: #{error.name}", error + attributes = {cause: error.name, step: 'create_sdp', type: z.calling.rtc.SDPType.ANSWER} + @call_et.telemetry.track_event z.tracking.EventName.CALLING.FAILED_RTC, undefined, attributes + + ### + Create a local SDP of type 'offer'. + + @see https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/createOffer + @private + @param restart [Boolean] Is ICE restart negotiation + ### + _create_offer: (restart) -> + offer_options = + iceRestart: restart + offerToReceiveAudio: true + offerToReceiveVideo: true + voiceActivityDetection: true + + @logger.log @logger.levels.INFO, "Creating '#{z.calling.rtc.SDPType.OFFER}' for flow with '#{@remote_user.name()}'" + @peer_connection.createOffer offer_options + .then (sdp_offer) => + @logger.log @logger.levels.DEBUG, "Creating '#{z.calling.rtc.SDPType.OFFER}' successful", sdp_offer + @telemetry.time_step z.telemetry.calling.CallSetupSteps.LOCAL_SDP_CREATED + @local_sdp @_rewrite_sdp sdp_offer, z.calling.enum.SDPSource.LOCAL + .catch (error) => + @logger.log @logger.levels.ERROR, "Creating '#{z.calling.rtc.SDPType.OFFER}' failed: #{error.name}", error + attributes = {cause: error.name, step: 'create_sdp', type: z.calling.rtc.SDPType.OFFER} + @call_et.telemetry.track_event z.tracking.EventName.CALLING.FAILED_RTC, undefined, attributes + @_solve_colliding_states() + + ### + Rewrite the SDP for compatibility reasons. + + @private + @param rtc_sdp [RTCSessionDescription] Session Description Protocol to be rewritten + @param sdp_source [z.calling.enum.SDPSource] Source of the SDP - local or remote + @return [RTCSessionDescription] Rewritten Session Description Protocol + ### + _rewrite_sdp: (rtc_sdp, sdp_source = z.calling.enum.SDPSource.REMOTE) -> + if sdp_source is z.calling.enum.SDPSource.LOCAL + rtc_sdp.sdp = rtc_sdp.sdp.replace 'UDP/TLS/', '' + + sdp_lines = rtc_sdp.sdp.split '\r\n' + outlines = [] + + ice_candidates = [] + + for sdp_line in sdp_lines + outline = sdp_line + + if sdp_line.startsWith 't=' + if sdp_source is z.calling.enum.SDPSource.LOCAL and not z.util.Environment.frontend.is_localhost() + outlines.push sdp_line + browser_string = "#{z.util.Environment.browser.name} #{z.util.Environment.browser.version}" + if z.util.Environment.electron + outline = "a=tool:electron #{z.util.Environment.version()} #{z.util.Environment.version false} (#{browser_string})" + else + outline = "a=tool:webapp #{z.util.Environment.version false} (#{browser_string})" + @logger.log @logger.levels.INFO, "Added tool version to local SDP: #{outline}" + + else if sdp_line.startsWith 'a=candidate' + ice_candidates.push sdp_line + + else if sdp_line.startsWith 'a=group' + if @negotiation_mode() is z.calling.enum.SDPNegotiationMode.STREAM_CHANGE and sdp_source is z.calling.enum.SDPSource.LOCAL + outlines.push 'a=x-streamchange' + @logger.log @logger.levels.INFO, 'Added stream renegotiation flag to local SDP' + + # Code to nail in bit-rate and ptime settings for improved performance and experience + else if sdp_line.startsWith 'm=audio' + if @negotiation_mode() is z.calling.enum.SDPNegotiationMode.ICE_RESTART or sdp_source is z.calling.enum.SDPSource.LOCAL and @is_group() + outlines.push sdp_line + outline = "b=AS:#{AUDIO_BITRATE}" + @logger.log @logger.levels.INFO, "Limited audio bit-rate in local SDP: #{outline}" + + else if sdp_line.startsWith 'm=video' + if sdp_source is z.calling.enum.SDPSource.LOCAL and @_should_rewrite_codecs() + outline = sdp_line.replace(' 98', '').replace ' 116', '' + @logger.log @logger.levels.WARN, 'Removed video codecs to prevent video freeze due to issue in Chrome 50 and 51' if outline isnt sdp_line + + else if sdp_line.startsWith 'a=fmtp' + if sdp_source is z.calling.enum.SDPSource.LOCAL and @_should_rewrite_codecs() + if sdp_line.endsWith '98 apt=116' + @logger.log @logger.levels.WARN, 'Removed FMTP line to prevent video freeze due to issue in Chrome 50 and 51' + outline = undefined + + else if sdp_line.startsWith 'a=rtpmap' + if sdp_source is z.calling.enum.SDPSource.LOCAL and @_should_rewrite_codecs() + if sdp_line.endsWith('98 rtx/90000') or sdp_line.endsWith '116 red/90000' + @logger.log @logger.levels.WARN, 'Removed RTPMAP line to prevent video freeze due to issue in Chrome 50 and 51' + outline = undefined + + if @negotiation_mode() is z.calling.enum.SDPNegotiationMode.ICE_RESTART or sdp_source is z.calling.enum.SDPSource.LOCAL and @is_group() + if z.util.contains sdp_line, 'opus' + outlines.push sdp_line + outline = "a=ptime:#{AUDIO_PTIME}" + @logger.log @logger.levels.INFO, "Changed audio p-time in local SDP: #{outline}" + + if outline isnt undefined + outlines.push outline + + @logger.log @logger.levels.INFO, + "'#{ice_candidates.length}' ICE candidate(s) found in '#{sdp_source}' SDP", ice_candidates + + rewritten_sdp = outlines.join '\r\n' + + if rtc_sdp.sdp isnt rewritten_sdp + rtc_sdp.sdp = rewritten_sdp + @logger.log @logger.levels.INFO, "Rewrote '#{sdp_source}' SDP of type '#{rtc_sdp.type}'", rtc_sdp + + return rtc_sdp + + ### + Initiates the sending of the local Session Description Protocol to the backend. + @private + ### + _send_local_sdp: -> + @local_sdp @_rewrite_sdp @peer_connection.localDescription, z.calling.enum.SDPSource.LOCAL + sdp_info = new z.calling.payloads.SDPInfo {conversation_id: @conversation_id, flow_id: @id, sdp: @local_sdp()} + + on_success = => + window.clearTimeout @send_sdp_timeout + @logger.log @logger.levels.INFO, "Sending local SDP of type '#{@local_sdp().type}' successful" + @telemetry.time_step z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SEND + @has_sent_local_sdp true + + on_failure = => + @logger.log @logger.levels.WARN, "Failed to send local SDP of type '#{@local_sdp().type}'" + + @logger.log @logger.levels.INFO, "Sending local SDP for flow with '#{@remote_user.name()}'\n#{@local_sdp().sdp}" + amplify.publish z.event.WebApp.CALL.SIGNALING.SEND_LOCAL_SDP_INFO, sdp_info, on_success, on_failure + + ### + Sets the local Session Description Protocol on the PeerConnection. + @private + ### + _set_local_sdp: -> + @logger.log @logger.levels.INFO, "Setting local SDP of type '#{@local_sdp().type}'", @local_sdp() + @peer_connection.setLocalDescription @local_sdp() + .then => + @logger.log @logger.levels.DEBUG, + "Setting local SDP of type '#{@local_sdp().type}' successful", @peer_connection.localDescription + @telemetry.time_step z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SET + @should_add_local_sdp false + @send_sdp_timeout = window.setTimeout => + @_send_local_sdp() if not @has_sent_local_sdp() + , 1000 + .catch (error) => + @logger.log @logger.levels.ERROR, "Setting local SDP of type '#{@local_sdp().type}' failed: #{error.name}", error + attributes = {cause: error.name, step: 'set_sdp', location: 'local', type: @local_sdp()?.type} + @call_et.telemetry.track_event z.tracking.EventName.CALLING.FAILED_RTC, undefined, attributes + + ### + Sets the remote Session Description Protocol on the PeerConnection. + @private + ### + _set_remote_sdp: -> + @logger.log @logger.levels.INFO, "Setting remote SDP of type '#{@remote_sdp().type}'\n#{@remote_sdp().sdp}" + @peer_connection.setRemoteDescription @remote_sdp() + .then => + @logger.log @logger.levels.DEBUG, + "Setting remote SDP of type '#{@remote_sdp().type}' successful", @peer_connection.remoteDescription + @telemetry.time_step z.telemetry.calling.CallSetupSteps.REMOTE_SDP_SET + @should_add_remote_sdp false + .catch (error) => + @logger.log @logger.levels.ERROR, "Setting remote SDP of type '#{@remote_sdp().type}' failed: #{error.name}", error + attributes = {cause: error.name, step: 'set_sdp', location: 'remote', type: @remote_sdp()?.type} + @call_et.telemetry.track_event z.tracking.EventName.CALLING.FAILED_RTC, undefined, attributes + + ### + Solve colliding SDP states. + @note If we receive a remote offer while we have a local offer, we need to check who needs to switch his SDP type. + @private + ### + _solve_colliding_states: -> + we_switched_state = false + if @self_user_id < @remote_user_id + @logger.log @logger.levels.WARN, + "We need to switch state of flow with '#{@remote_user.name()}'. Local SDP needs to be changed." + we_switched_state = true + @_switch_local_sdp_state() + else + @logger.log @logger.levels.WARN, + "Remote side needs to switch state of flow with '#{@remote_user.name()}'. Waiting for new remote SDP." + + return we_switched_state + + + ############################################################################### + # SDP sate collision handling + ############################################################################### + + ### + Switch the local SDP state. + @note Set converted flag first, because it influences the tear-down of the PeerConnection + @private + ### + _switch_local_sdp_state: -> + @logger.log @logger.levels.DEBUG, '_switch_local_sdp_state' + + @logger.log @logger.levels.LEVEL_2, + "Switching from local #{@local_sdp_type()} to local '#{z.calling.rtc.SDPType.ANSWER}'" + + @converted_own_sdp_state true + @is_answer true + remote_sdp_cache = @remote_sdp() + @_close_peer_connection() + @_reset_signaling_states() + @_initialize_peer_connection() + @remote_sdp remote_sdp_cache + @_set_remote_sdp() + @converted_own_sdp_state false + + + ############################################################################### + # ICE candidate handling + ############################################################################### + + ### + Add or cache remote ICE candidate. + @param ice_candidate [RTCIceCandidate] Received remote ICE candidate + ### + add_remote_ice_candidate: (ice_candidate) => + if z.util.contains ice_candidate.candidate, 'end-of-candidates' + @logger.log @logger.levels.INFO, 'Ignoring remote non-candidate' + return + + if @can_set_ice_candidates() + @_add_ice_candidate ice_candidate + @ice_candidates_cache.push ice_candidate + else + @ice_candidates_cache.push ice_candidate + @logger.log @logger.levels.INFO, 'Cached ICE candidate for flow' + + ### + Add all cached ICE candidates to the flow. + @private + ### + _add_cached_ice_candidates: -> + if @ice_candidates_cache.length + @logger.log @logger.levels.INFO, "Adding '#{@ice_candidates_cache.length}' cached ICE candidates" + @_add_ice_candidate ice_candidate for ice_candidate in @ice_candidates_cache + else + @logger.log @logger.levels.INFO, 'No cached ICE candidates found' + + ### + Add a remote ICE candidate to the flow directly. + @private + @param ice_candidate [RTCICECandidate] Received remote ICE candidate + ### + _add_ice_candidate: (ice_candidate) -> + @logger.log @logger.levels.INFO, "Adding ICE candidate to flow with '#{@remote_user.name()}'", ice_candidate + @peer_connection.addIceCandidate ice_candidate + .then => + @logger.log @logger.levels.DEBUG, 'Adding ICE candidate successful' + .catch (error) => + @logger.log @logger.levels.WARN, "Adding ICE candidate failed: #{error.name}", error + attributes = {cause: error.name, step: 'add_candidate', type: z.calling.rtc.SDPType.OFFER} + @call_et.telemetry.track_event z.tracking.EventName.CALLING.FAILED_RTC, undefined, attributes + + ### + Create a fake ICE candidate from a message. + @param candidate_message [String] Candidate message for the RTCICECandidate + @return [Object] Object containing data for RTCICECandidate + ### + _fake_ice_candidate: (candidate_message) -> + return {} = + candidate: candidate_message + sdpMLineIndex: 0 + sdpMid: 'audio' + + ### + Send an ICE candidate to the backend. + @private + @param ice_candidate [RTCICECandidate] Local ICE candidate to be send + ### + _send_ice_candidate: (ice_candidate) -> + if not z.util.contains ice_candidate.candidate, 'UDP' + return @logger.log @logger.levels.INFO, "Local ICE candidate ignored as it is not of type 'UDP'" + + if @conversation_id and @id + ice_info = new z.calling.payloads.ICECandidateInfo @conversation_id, @id, ice_candidate + @logger.log @logger.levels.INFO, 'Sending ICE candidate', ice_info + amplify.publish z.event.WebApp.CALL.SIGNALING.SEND_ICE_CANDIDATE_INFO, ice_info + + ### + Should a local SDP be rewritten to prevent frozen video. + @note All sections that rewrite the SDP for this can be removed once we require Chrome 52 + @return [Boolean] Should SDP be rewritten + ### + _should_rewrite_codecs: -> + return z.util.Environment.browser.requires.calling_codec_rewrite + + + ############################################################################### + # Media stream handling + ############################################################################### + + ### + Inject an audio file into the flow. + @param audio_file_path [String] Path to the audio file + @param callback [Function] Function to be called when completed + ### + inject_audio_file: (audio_file_path, callback) => + @audio.inject_audio_file audio_file_path, callback + + ### + Switch out the local MediaStream. + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Object containing the required MediaStream information + @return [Promise] Promise that resolves when the updated MediaStream is used + ### + switch_media_stream: (media_stream_info) => + if @peer_connection.getSenders? + @_replace_media_track media_stream_info + .then -> + return [media_stream_info, false] + else + @_replace_media_stream media_stream_info + .then (media_stream_info) => + @is_answer false + return [media_stream_info, true] + + ### + Adds a local MediaStream to the PeerConnection. + @private + @param media_stream [MediaStream] MediaStream to add to the PeerConnection + ### + _add_media_stream: (media_stream) => + if media_stream.type is z.calling.enum.MediaType.AUDIO + @peer_connection.addStream @audio.wrap_microphone_stream media_stream + else + @peer_connection.addStream media_stream + @logger.log @logger.levels.INFO, "Added local MediaStream of type '#{media_stream.type}' to PeerConnection", + {stream: media_stream, audio_tracks: media_stream.getAudioTracks(), video_tracks: media_stream.getVideoTracks()} + + ### + Adds the local MediaStreams to the PeerConnection. + @private + ### + _add_media_streams: -> + media_streams_identical = @_compare_local_media_streams() + + @_add_media_stream @audio_stream() if @audio_stream() + @_add_media_stream @video_stream() if @video_stream() and not media_streams_identical + + ### + Compare whether local audio and video streams are identical. + @private + ### + _compare_local_media_streams: -> + return @audio_stream() and @video_stream() and @audio_stream().id is @video_stream().id + + ### + Replace the MediaStream attached to the PeerConnection. + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Object containing the required MediaStream information + ### + _replace_media_stream: (media_stream_info) => + Promise.resolve @_remove_media_streams media_stream_info.type + .then => + @negotiation_mode z.calling.enum.SDPNegotiationMode.STREAM_CHANGE + return @_upgrade_media_stream media_stream_info + .then (media_stream_info) => + @_add_media_stream media_stream_info.stream + @logger.log @logger.levels.INFO, 'Replaced the MediaStream successfully', media_stream_info.stream + return media_stream_info + + ### + Replace the a MediaStreamTrack attached to the MediaStream of the PeerConnection. + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Object containing the required MediaStream information + ### + _replace_media_track: (media_stream_info) => + media_stream_track = media_stream_info.stream.getTracks()[0] + return Promise.all (sender.replaceTrack media_stream_track for sender in @peer_connection.getSenders() when sender.track.kind is media_stream_info.type) + .then => + @logger.log @logger.levels.INFO, "Replaced the '#{media_stream_info.type}' track" + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to replace the '#{media_stream_info.type}' track: #{error.message}", error + + ### + Reset the flows MediaStream and media elements. + @private + @param media_stream [MediaStream] Local MediaStream to remove from the PeerConnection + ### + _remove_media_stream: (media_stream) => + if @peer_connection + try + if @peer_connection.signalingState isnt z.calling.rtc.SignalingState.CLOSED + @peer_connection.removeStream media_stream + @logger.log @logger.levels.INFO, "Removed local MediaStream of type '#{media_stream.type}' from PeerConnection", + {stream: media_stream, audio_tracks: media_stream.getAudioTracks(), video_tracks: media_stream.getVideoTracks()} + # @param [InvalidStateError] error + catch error + @logger.log @logger.levels.ERROR, "We caught the #{error.message}", error + Raygun.send new Error('Failed to remove MediaStream from PeerConnection'), error + else + @logger.log @logger.levels.INFO, 'No PeerConnection found to remove MediaStream from' + + ### + Reset the flows MediaStream and media elements. + @private + @param media_type [z.calling.enum.MediaType] Optional media type of MediaStreams to be removed + ### + _remove_media_streams: (media_type = z.calling.enum.MediaType.AUDIO_VIDEO) => + switch media_type + when z.calling.enum.MediaType.AUDIO_VIDEO + media_streams_identical = @_compare_local_media_streams() + + @_remove_media_stream @audio_stream() if @audio_stream() + @_remove_media_stream @video_stream() if @video_stream() and not media_streams_identical + when z.calling.enum.MediaType.AUDIO + @_remove_media_stream @audio_stream() if @audio_stream() + when z.calling.enum.MediaType.VIDEO + @_remove_media_stream @video_stream() if @video_stream() + + ### + Upgrade a MediaStream with missing audio or video. + @private + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Contains the info about the MediaStream to be updated + @return [z.calling.payloads.MediaStreamInfo] + ### + _upgrade_media_stream: (media_stream_info) -> + if media_stream_info.type is z.calling.enum.MediaType.AUDIO and @video_stream() + media_stream_tracks = z.calling.handler.MediaStreamHandler.get_media_tracks @video_stream(), z.calling.enum.MediaType.VIDEO + + else if media_stream_info.type is z.calling.enum.MediaType.VIDEO and @audio_stream() + media_stream_tracks = z.calling.handler.MediaStreamHandler.get_media_tracks @audio_stream(), z.calling.enum.MediaType.AUDIO + + if media_stream_tracks?.length + @audio_stream().removeTrack media_stream_tracks[0] if @audio_stream() + @video_stream().removeTrack media_stream_tracks[0] if @video_stream() + media_stream_info.stream.addTrack media_stream_tracks[0] + @logger.log @logger.levels.INFO, "Upgraded local MediaStream of type '#{media_stream_info.type}' with '#{media_stream_tracks[0].kind}'", + {stream: media_stream_info.stream, audio_tracks: media_stream_info.stream.getAudioTracks(), video_tracks: media_stream_info.stream.getVideoTracks()} + media_stream_info.update_stream_type() + else + @logger.log @logger.levels.INFO, 'No changes to the new local MediaStream', + {stream: media_stream_info.stream, audio_tracks: media_stream_info.stream.getAudioTracks(), video_tracks: media_stream_info.stream.getVideoTracks()} + + return media_stream_info + + + ############################################################################### + # Reset + ############################################################################### + + ### + Reset the flow. + @note Reset PC initialized first to prevent new local SDP + ### + reset_flow: => + @logger.log @logger.levels.INFO, "Resetting flow '#{@id}'" + @telemetry.reset_statistics() + .then (statistics) => + @logger.log @logger.levels.INFO, 'Flow network stats updated for the last time', statistics + amplify.publish z.event.WebApp.DEBUG.UPDATE_LAST_CALL_STATUS, @telemetry.create_report() + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to reset flow networks stats: #{error.message}" + try + if @peer_connection?.signalingState isnt z.calling.rtc.SignalingState.CLOSED + @_close_peer_connection() + catch error + @logger.log @logger.levels.ERROR, "We caught the #{error.name}", error + @_remove_media_streams() + @_reset_signaling_states() + @ice_candidates_cache = [] + @payload undefined + @pc_initialized false + @logger.log @logger.levels.DEBUG, "Resetting flow '#{@id}' with user '#{@remote_user.name()}' successful" + + ### + Reset the signaling states. + @private + ### + _reset_signaling_states: -> + @signaling_state z.calling.rtc.SignalingState.NEW + @connection_state z.calling.rtc.ICEConnectionState.NEW + @gathering_state z.calling.rtc.ICEGatheringState.NEW + + + ############################################################################### + # Logging + ############################################################################### + + # Get full telemetry report. + get_telemetry: => + @telemetry.get_report() + + # Log flow status to console. + log_status: => + @telemetry.log_status @participant_et + + # Log flow setup step timings to console. + log_timings: => + @telemetry.log_timings() + + # Report flow status to Raygun. + report_status: => + @telemetry.report_status() + + # Report flow setup step timings to Raygun. + report_timings: => + @telemetry.report_timings() diff --git a/app/script/calling/entities/FlowAudio.coffee b/app/script/calling/entities/FlowAudio.coffee new file mode 100644 index 00000000000..d16710d52e5 --- /dev/null +++ b/app/script/calling/entities/FlowAudio.coffee @@ -0,0 +1,120 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.entities ?= {} + +class z.calling.entities.FlowAudio + constructor: (@flow_et, @audio_context) -> + @logger = new z.util.Logger "z.calling.FlowAudio (#{@flow_et.id})", z.config.LOGGER.OPTIONS + + # Panning + @panning = @flow_et.participant_et.panning + @panning.subscribe (new_value) => + @logger.log @logger.levels.INFO, "Panning of #{@flow_et.remote_user.name()} changed to '#{new_value}'" + @set_pan new_value + + @pan_node = undefined + @gain_node = undefined + @audio_source = undefined + @audio_remote = undefined + + amplify.subscribe z.event.WebApp.CALL.MEDIA.MUTE_AUDIO, @set_gain_node + + ### + @param is_active [Boolean] Whether the flow is active + ### + hookup: (is_active) => + if is_active is true + @_hookup_audio() + else + @audio_source.disconnect() if @audio_source? + + inject_audio_file: (audio_file_path, callback) => + return if not @audio_context? + + # Load audio file + request = new XMLHttpRequest() + request.open 'GET', audio_file_path, true + request.responseType = 'arraybuffer' + request.onload = => + load = (buffer) => + @logger.log @logger.levels.INFO, "Loaded audio from '#{audio_file_path}'" + # Play audio file + audio_buffer = buffer + file_source = @audio_context.createBufferSource() + file_source.buffer = audio_buffer + @audio_source.disconnect() + file_source.connect @audio_remote + file_source.onended = => + @logger.log @logger.levels.INFO, 'Finished playing audio file' + file_source.disconnect @audio_remote + @_hookup_audio() + + if callback? + @logger.log @logger.levels.INFO, 'Invoking callback after playing audio file' + callback() + + @logger.log @logger.levels.INFO, 'Playing audio file' + file_source.start() + fail = => + @logger.log @logger.levels.ERROR, "Failed to load audio from '#{audio_file_path}'" + @audio_context.decodeAudioData request.response, load, fail + request.send() + + set_gain_node: (is_muted) => + if @gain_node + if is_muted + @gain_node.gain.value = 0 + else + @gain_node.gain.value = 1 + @logger.log @logger.levels.INFO, "Outgoing audio on flow muted '#{is_muted}'" + + set_pan: (panning_value) => + if @pan_node + @pan_node.pan.value = panning_value + + wrap_microphone_stream: (media_stream) => + wrapped_stream = media_stream + if @audio_context + @audio_source = @audio_context.createMediaStreamSource media_stream + @gain_node = @audio_context.createGain() + @audio_remote = @audio_context.createMediaStreamDestination() + @_hookup_audio() + wrapped_stream = @audio_remote.stream + @logger.log @logger.levels.INFO, 'Wrapped audio stream from microphone', wrapped_stream + return wrapped_stream + + wrap_speaker_stream: (media_stream) => + wrapped_stream = media_stream + if z.util.Environment.browser.firefox + if @audio_context + remote_source = @audio_context.createMediaStreamSource media_stream + @pan_node = @audio_context.createStereoPanner() + @pan_node.pan.value = @panning() + speaker = @audio_context.createMediaStreamDestination() + remote_source.connect @pan_node + @pan_node.connect speaker + wrapped_stream = speaker.stream + @logger.log @logger.levels.INFO, "Wrapped audio stream to speaker to create stereo. Initial panning set to '#{@panning()}'.", wrapped_stream + return wrapped_stream + + _hookup_audio: => + @audio_source.connect @gain_node if @audio_source + @gain_node.connect @audio_remote if @gain_node diff --git a/app/script/calling/entities/Participant.coffee b/app/script/calling/entities/Participant.coffee new file mode 100644 index 00000000000..6c9be11f283 --- /dev/null +++ b/app/script/calling/entities/Participant.coffee @@ -0,0 +1,58 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.entities ?= {} + +# Participant entity. +class z.calling.entities.Participant + ### + Construct a new participant. + @param user [z.entity.User] User entity to base the participant on + ### + constructor: (@user) -> + @flow = ko.observable() + @is_connected = ko.observable false + @panning = ko.observable 0.0 + @was_connected = false + + @state = + muted: ko.observable false + screen_shared: ko.observable false + videod: ko.observable false + + @is_connected.subscribe (is_connected) -> + if is_connected and not @was_connected + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.READY_TO_TALK + @was_connected = true + + ### + Add a new flow to the participant. + @param flow_et [z.calling.Flow] Flow entity to be added to the flow + ### + add_flow: (flow_et) => + @flow flow_et unless @flow()?.id is flow_et.id + + + ### + Get the flow of the participant. + @return [z.calling.Flow] Flow entity of participant + ### + get_flow: => + return @flow() diff --git a/app/script/calling/enum/CallFinishedReason.coffee b/app/script/calling/enum/CallFinishedReason.coffee new file mode 100644 index 00000000000..ef384930da4 --- /dev/null +++ b/app/script/calling/enum/CallFinishedReason.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.CallFinishedReason = + CONNECTION_DROPPED: 'connection_dropped' + COMPLETED: 'completed' + MISSED: 'missed' + OTHER_USER: 'other_user' + SELF_USER: 'self_user' + UNKNOWN: 'unknown' diff --git a/app/script/calling/enum/CallState.coffee b/app/script/calling/enum/CallState.coffee new file mode 100644 index 00000000000..a03ea642feb --- /dev/null +++ b/app/script/calling/enum/CallState.coffee @@ -0,0 +1,31 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.CallState = + CANCELED: 'canceled' + CONNECTING: 'connecting' + DELETED: 'deleted' + IGNORED: 'ignored' + INCOMING: 'incoming' + ONGOING: 'ongoing' + OUTGOING: 'outgoing' + UNKNOWN: 'unknown' diff --git a/app/script/calling/enum/CallStateEventCause.coffee b/app/script/calling/enum/CallStateEventCause.coffee new file mode 100644 index 00000000000..01c29327abd --- /dev/null +++ b/app/script/calling/enum/CallStateEventCause.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.CallStateEventCause = + REQUESTED: 'requested' # Voluntarily by client through the API + DISCONNECTED: 'disconnected' # WebSocket disconnected + INTERRUPTED: 'interrupted' # Involuntarily by client through the API (e.g. GSM) + GONE: 'gone' diff --git a/app/script/calling/enum/CallStateGroups.coffee b/app/script/calling/enum/CallStateGroups.coffee new file mode 100644 index 00000000000..7451a85f963 --- /dev/null +++ b/app/script/calling/enum/CallStateGroups.coffee @@ -0,0 +1,43 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.CallStateGroups = + IS_ACTIVE: [ + z.calling.enum.CallState.CONNECTING + z.calling.enum.CallState.INCOMING + z.calling.enum.CallState.ONGOING + z.calling.enum.CallState.OUTGOING + ] + IS_ENDED: [ + z.calling.enum.CallState.DELETED + z.calling.enum.CallState.UNKNOWN + ] + IS_RINGING: [ + z.calling.enum.CallState.INCOMING + z.calling.enum.CallState.OUTGOING + ] + CAN_CONNECT: [ + z.calling.enum.CallState.IGNORED + z.calling.enum.CallState.INCOMING + z.calling.enum.CallState.ONGOING + z.calling.enum.CallState.OUTGOING + ] diff --git a/app/script/calling/enum/MediaDeviceType.coffee b/app/script/calling/enum/MediaDeviceType.coffee new file mode 100644 index 00000000000..b1855ee054e --- /dev/null +++ b/app/script/calling/enum/MediaDeviceType.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.MediaDeviceType = + AUDIO_INPUT: 'audioinput' + AUDIO_OUTPUT: 'audiooutput' + SCREEN_INPUT: 'screeninput' + VIDEO_INPUT: 'videoinput' diff --git a/app/script/calling/enum/MediaStreamSource.coffee b/app/script/calling/enum/MediaStreamSource.coffee new file mode 100644 index 00000000000..62de1868d2f --- /dev/null +++ b/app/script/calling/enum/MediaStreamSource.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.MediaStreamSource = + LOCAL: 'local' + REMOTE: 'remote' diff --git a/app/script/calling/enum/MediaType.coffee b/app/script/calling/enum/MediaType.coffee new file mode 100644 index 00000000000..7ca71d4606c --- /dev/null +++ b/app/script/calling/enum/MediaType.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.MediaType = + AUDIO: 'audio' + AUDIO_VIDEO: 'audio/video' + NONE: 'none' + SCREEN: 'screen' + VIDEO: 'video' diff --git a/app/script/calling/enum/ParticipantState.coffee b/app/script/calling/enum/ParticipantState.coffee new file mode 100644 index 00000000000..0afce09bf7b --- /dev/null +++ b/app/script/calling/enum/ParticipantState.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.ParticipantState = + IDLE: 'idle' + JOINED: 'joined' diff --git a/app/script/calling/enum/SDPNegotiationMode.coffee b/app/script/calling/enum/SDPNegotiationMode.coffee new file mode 100644 index 00000000000..cafd6b34a83 --- /dev/null +++ b/app/script/calling/enum/SDPNegotiationMode.coffee @@ -0,0 +1,26 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.SDPNegotiationMode = + DEFAULT: 'default' + ICE_RESTART: 'ice_restart' + STREAM_CHANGE: 'stream_change' diff --git a/app/script/calling/enum/SDPSource.coffee b/app/script/calling/enum/SDPSource.coffee new file mode 100644 index 00000000000..4fe3f2f8985 --- /dev/null +++ b/app/script/calling/enum/SDPSource.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.SDPSource = + LOCAL: 'local' + REMOTE: 'remote' diff --git a/app/script/calling/enum/VideoOrientation.coffee b/app/script/calling/enum/VideoOrientation.coffee new file mode 100644 index 00000000000..a19ddf27687 --- /dev/null +++ b/app/script/calling/enum/VideoOrientation.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.enum ?= {} + +z.calling.enum.VideoOrientation = + LANDSCAPE: 'landscape' + PORTRAIT: 'portrait' diff --git a/app/script/calling/handler/CallSignalingHandler.coffee b/app/script/calling/handler/CallSignalingHandler.coffee new file mode 100644 index 00000000000..1cc804ba7f0 --- /dev/null +++ b/app/script/calling/handler/CallSignalingHandler.coffee @@ -0,0 +1,333 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.handler ?= {} + +# Call signaling handler +class z.calling.handler.CallSignalingHandler + ### + Construct a new call signaling handler. + @param call_center [z.calling.CallCenter] Call center with references to all other handlers + ### + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.calling.handler.CallSignalingHandler', z.config.LOGGER.OPTIONS + + # Caches + @candidate_cache = {} + @sdp_cache = {} + + # Mapper + @ice_mapper = new z.calling.mapper.ICECandidateMapper() + @sdp_mapper = new z.calling.mapper.SDPMapper() + + @subscribe_to_events() + return + + # Subscribe to amplify topics. + subscribe_to_events: => + amplify.subscribe z.event.WebApp.CALL.SIGNALING.DELETE_FLOW, @delete_flow + amplify.subscribe z.event.WebApp.CALL.SIGNALING.POST_FLOWS, @post_for_flows + amplify.subscribe z.event.WebApp.CALL.SIGNALING.SEND_ICE_CANDIDATE_INFO, @post_ice_candidate + amplify.subscribe z.event.WebApp.CALL.SIGNALING.SEND_LOCAL_SDP_INFO, @put_local_sdp + + # Un-subscribe from amplify topics. + un_subscribe: -> + subscriptions = [ + z.event.WebApp.CALL.SIGNALING.DELETE_FLOW + z.event.WebApp.CALL.SIGNALING.POST_FLOWS + z.event.WebApp.CALL.SIGNALING.SEND_ICE_CANDIDATE_INFO + z.event.WebApp.CALL.SIGNALING.SEND_LOCAL_SDP_INFO + ] + amplify.unsubscribeAll topic for topic in subscriptions + + + ############################################################################### + # Events + ############################################################################### + + ### + Handling of 'call.flow-add' events. + @param event [Object] Event payload + ### + on_flow_add_event: (event) => + @call_center.get_call_by_id event.conversation + .then (call_et) => + @_add_flow call_et, flow for flow in event.flows + .catch (error) => + @logger.log @logger.levels.WARN, "Ignored 'call.flow-add' in '#{event.conversation}' that has no call", error + + ### + Handling of 'call.flow-delete' events. + @param event [Object] Event payload + ### + on_flow_delete_event: (event) => + @call_center.get_call_by_id event.conversation + .then (call_et) => + @_add_flow call_et, event.flow + .catch => + @logger.log @logger.levels.WARN, "Ignored 'call.flow-delete' in '#{event.conversation}' that has no call", event + + ### + Handling of 'call.remote-candidates-add' and 'call.remote-candidates-update' events. + @param event [Object] Event payload + ### + on_remote_ice_candidates: (event) => + mapped_candidates = (@ice_mapper.map_ice_message_to_object candidate for candidate in event.candidates) + + @call_center.get_call_by_id event.conversation + .then (call_et) => + # And either add + if flow_et = call_et.get_flow_by_id event.flow + @logger.log @logger.levels.INFO, "Received '#{mapped_candidates.length}' ICE candidates for existing flow '#{event.flow}'", mapped_candidates + for ice_candidate in mapped_candidates + flow_et.add_remote_ice_candidate ice_candidate + else + throw new z.calling.CallError "Flow '#{event.flow}' not found", z.calling.CallError::TYPE.FLOW_NOT_FOUND + .catch => + # Or cache them + @logger.log @logger.levels.INFO, "Cached '#{mapped_candidates.length}' ICE candidates for unknown flow '#{event.flow}'", mapped_candidates + for ice_candidate in mapped_candidates + @_cache_ice_candidate event.flow, ice_candidate + + ### + Handling of 'call.remote-sdp' events. + @param event [Object] Event payload + ### + on_remote_sdp: (event) => + remote_sdp = @sdp_mapper.map_sdp_event_to_object event + + @call_center.get_call_by_id event.conversation + .then (call_et) => + if flow_et = call_et.get_flow_by_id event.flow + @logger.log @logger.levels.INFO, "Received remote SDP for existing flow '#{event.flow}'", remote_sdp + flow_et.save_remote_sdp remote_sdp + else + throw new z.calling.CallError "Flow '#{event.flow}' not found", z.calling.CallError::TYPE.FLOW_NOT_FOUND + .catch => + if event.state is z.calling.rtc.SDPType.OFFER + @_cache_remote_sdp event.flow, remote_sdp + @logger.log @logger.levels.INFO, "Cached remote SDP for unknown flow '#{event.flow}'", remote_sdp + else + @logger.log @logger.levels.WARN, "Ignored remote SDP non-offer before call for flow '#{event.flow}'", remote_sdp + + + ############################################################################### + # Flows + ############################################################################### + + ### + Delete a flow on the backend. + @private + @param delete_flow_info [z.calling.payloads.FlowDeletionInfo] Contains Conversation ID, Flow ID and Reason for flow deletion + ### + delete_flow: (flow_info) => + Promise.resolve @call_center.media_element_handler.remove_media_element flow_info.flow_id + .then => + @logger.log @logger.levels.INFO, "DELETEing flow '#{flow_info.flow_id}'" + return @call_center.call_service.delete_flow flow_info, [] + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request flow_info.conversation_id, jqXHR + @logger.log @logger.levels.DEBUG, "DELETEing flow '#{flow_info.flow_id}' successful" + if flow_info.reason is z.calling.payloads.FlowDeletionReason.RELEASED + @logger.log @logger.levels.DEBUG, 'Flow was released - We need implement posting for flows to renegotiate' + return response + .catch (error) => + if error.label is z.service.BackendClientError::LABEL.IN_USE + @logger.log @logger.levels.WARN, "DELETEing flow '#{flow_info.flow_id}' would have to be forced" + flow_info.reason = z.calling.payloads.FlowDeletionReason.RELEASED + return @delete_flow flow_info + else + @logger.log @logger.levels.ERROR, "DELETEing flow '#{flow_info.flow_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'delete', request: 'flows'} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + + ### + Post for flows. + @param conversation_id [String] Conversation ID of call to be posted for flows + ### + post_for_flows: (conversation_id) => + @logger.log @logger.levels.INFO, "POSTing for flows in conversation '#{conversation_id}'" + @call_center.call_service.post_flows conversation_id, [] + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request conversation_id, jqXHR + return @call_center.get_call_by_id conversation_id + .then (call_et) => + @logger.log @logger.levels.DEBUG, "POSTing for flows in '#{conversation_id}' successful", response + @_add_flow call_et, flow for flow in response.flows when flow.active is true + .catch (error) => + if error.type is z.calling.CallError::TYPE.CALL_NOT_FOUND + @logger.log @logger.levels.WARN, "POSTing for flows in '#{conversation_id}' successful, call gone", error + else + @logger.log @logger.levels.ERROR, + "POSTing for flows in conversation '#{conversation_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'post', request: 'flows'} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + + ### + Create a flow in a call. + + @private + @param call_et [z.calling.Call] Call entity + @param payload [Object] Payload for call to be created + ### + _add_flow: (call_et, payload) -> + @call_center.user_repository.get_user_by_id payload.remote_user, (user_et) => + # Get or construct flow entity + flow_et = call_et.get_flow_by_id payload.id + if not flow_et + flow_et = call_et.construct_flow payload.id, user_et, @call_center.audio_repository.get_audio_context(), @call_center.timings() + + # Add payload to flow entity + flow_et.add_payload payload + + # Unpack cache entries + if @sdp_cache[flow_et.id] isnt undefined + flow_et.save_remote_sdp @sdp_cache[flow_et.id] + delete @sdp_cache[flow_et.id] + if @candidate_cache[flow_et.id] isnt undefined + for ice_candidate in @candidate_cache[flow_et.id] + flow_et.add_remote_ice_candidate ice_candidate + delete @candidate_cache[flow_et.id] + + ### + Delete all flows from a call. + @private + @param conversation_id [String] Conversation ID to get and delete all flows for + ### + _delete_flows: (conversation_id) -> + @logger.log @logger.levels.WARN, "Deleting all flows for '#{conversation_id}'" + @_get_flows conversation_id + .then (flows) => + flows_to_delete = flows.length + + if flows_to_delete is 0 + @logger.log @logger.levels.INFO, "No flows for conversation '#{conversation_id}' to delete" + return + + @logger.log @logger.levels.INFO, "We will cleanup '#{flows.length}' flows from conversation '#{conversation_id}'" + + deletions = 0 + for flow in flows + flow_deletion_info = new z.calling.payloads.FlowDeletionInfo conversation_id, flow.id + @delete_flow flow_deletion_info + .then => + deletions += 1 + if deletions is flows_to_delete + @logger.log @logger.levels.INFO, "We deleted all '#{deletions}' flows for conversation '#{conversation_id}'" + + ### + Get flows from backend. + @private + @param conversation_id [String] Conversation ID of call to get flows for + ### + _get_flows: (conversation_id) -> + @logger.log @logger.levels.INFO, "GETting flows for '#{conversation_id}'" + return @call_center.call_service.get_flows conversation_id, [] + .then (response_array) => + [response, jqXHR] = response_array + @logger.log @logger.levels.DEBUG, "GETting flows for '#{conversation_id}' successful" + return response.flows + .catch (error) => + @logger.log @logger.levels.ERROR, "GETting flows for '#{conversation_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'get', request: 'flows'} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + + + ############################################################################### + # SDP handling + ############################################################################### + + ### + Put the local SDP on the backend. + + @param sdp_info [z.calling.payloads.SDPInfo] SDP info to be send + @param on_success [Function] Function to be called on success + @param on_error [Function] Function to be called on failure + ### + put_local_sdp: (sdp_info, on_success, on_failure) => + @logger.log @logger.levels.INFO, "PUTting local SDP for flow '#{sdp_info.flow_id}'", sdp_info + @call_center.call_service.put_local_sdp sdp_info.conversation_id, sdp_info.flow_id, sdp_info.sdp, [] + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request sdp_info.conversation_id, jqXHR + @logger.log @logger.levels.DEBUG, "PUTting local SDP for flow '#{sdp_info.flow_id}' successful" + on_success? response + .catch (error) => + @logger.log @logger.levels.ERROR, + "PUTting local SDP for flow '#{sdp_info.flow_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'put', request: 'sdp', sdp_type: sdp_info.sdp.type} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + on_failure? new Error error + + ### + Cache remote SDP until we have the flow. + + @private + @param flow_id [String] Flow ID + @param sdp [RTCSessionDescription] Remote SDP + ### + _cache_remote_sdp: (flow_id, sdp) -> + @sdp_cache[flow_id] = sdp + window.setTimeout => + delete @sdp_cache[flow_id] + , 60000 + + + ############################################################################### + # ICE candidate handling + ############################################################################### + + ### + Post a local ICE candidate to the backend. + @param ice_info [z.calling.payloads.ICECandidateInfo] ICE candidate info to be send + ### + post_ice_candidate: (ice_info) => + candidate = @ice_mapper.map_ice_object_to_message ice_info.ice_candidate + + @logger.log @logger.levels.INFO, "POSTing local ICE candidate for flow '#{ice_info.flow_id}'", candidate + @call_center.call_service.post_local_candidates ice_info.conversation_id, ice_info.flow_id, candidate, [] + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request ice_info.conversation_id, jqXHR + @logger.log @logger.levels.INFO, "POSTing local ICE candidate for flow '#{ice_info.flow_id}' successful" + .catch (error) => + @logger.log @logger.levels.ERROR, + "POSTing local ICE candidate for flow '#{ice_info.flow_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'put', request: 'ice_candidate'} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + + ### + Cache remote ICE candidate until we have the flow. + + @private + @param flow_id [String] Flow ID + @param candidate [RTCIceCandidate] Remote ICE candidate + ### + _cache_ice_candidate: (flow_id, candidate) -> + list = @candidate_cache[flow_id] + if list is undefined + list = [] + @candidate_cache[flow_id] = list + window.setTimeout => + delete @candidate_cache[flow_id] + , 60000 + list.push candidate diff --git a/app/script/calling/handler/CallStateHandler.coffee b/app/script/calling/handler/CallStateHandler.coffee new file mode 100644 index 00000000000..1621efbefc0 --- /dev/null +++ b/app/script/calling/handler/CallStateHandler.coffee @@ -0,0 +1,748 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.handler ?= {} + +# Call state handler +class z.calling.handler.CallStateHandler + ### + Construct a new call state handler. + @param call_center [z.calling.CallCenter] Call center with references to all other handlers + ### + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.calling.handler.CallStateHandler', z.config.LOGGER.OPTIONS + + @calls = ko.observableArray [] + @joined_call = ko.pureComputed => return call_et for call_et in @calls() when call_et.self_client_joined() + + @self_state = + muted: @call_center.media_stream_handler.self_stream_state.muted + screen_shared: @call_center.media_stream_handler.self_stream_state.screen_shared + videod: @call_center.media_stream_handler.self_stream_state.videod + + @is_handling_notifications = ko.observable true + @subscribe_to_events() + return + + # Subscribe to amplify topics. + subscribe_to_events: => + amplify.subscribe z.event.WebApp.CALL.STATE.CHECK, @check_state + amplify.subscribe z.event.WebApp.CALL.STATE.DELETE, @delete_call + amplify.subscribe z.event.WebApp.CALL.STATE.IGNORE, @ignore_call + amplify.subscribe z.event.WebApp.CALL.STATE.JOIN, @join_call + amplify.subscribe z.event.WebApp.CALL.STATE.LEAVE, @leave_call + amplify.subscribe z.event.WebApp.CALL.STATE.REMOVE_PARTICIPANT, @remove_participant + amplify.subscribe z.event.WebApp.CALL.STATE.TOGGLE, @toggle_joined + amplify.subscribe z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, @set_notification_handling_state + + # Un-subscribe from amplify topics. + un_subscribe: -> + subscriptions = [ + z.event.WebApp.CALL.STATE.CHECK + z.event.WebApp.CALL.STATE.DELETE + z.event.WebApp.CALL.STATE.JOIN + z.event.WebApp.CALL.STATE.LEAVE + z.event.WebApp.CALL.STATE.REMOVE_PARTICIPANT + z.event.WebApp.CALL.STATE.TOGGLE + ] + amplify.unsubscribeAll topic for topic in subscriptions + + + ############################################################################### + # Notification stream handling + ############################################################################### + + ### + Check for ongoing call in conversation. + @param conversation_id [String] Conversation ID + ### + check_state: (conversation_id) => + conversation_et = @call_center.conversation_repository.get_conversation_by_id conversation_id + return if not conversation_et? or conversation_et.removed_from_conversation() + + @_is_call_ongoing conversation_id + .then ([is_call_ongoing, response]) => + if is_call_ongoing + @on_call_state @_fake_on_state_event(response, conversation_id), true + @call_center.conversation_repository.unarchive_conversation conversation_et if conversation_et.is_archived() + + ### + Set the notification handling state. + @note Temporarily ignore call related events when handling notifications from the stream + @param handling_notifications [Boolean] Notifications are being handled from the stream + ### + set_notification_handling_state: (handling_notifications) => + @is_handling_notifications handling_notifications + @_update_ongoing_calls() if not @is_handling_notifications() + @logger.log @logger.levels.INFO, "Ignoring call events: #{handling_notifications}" + + ### + Update state of currently ongoing calls. + @private + ### + _update_ongoing_calls: -> + for call_et in @calls() + @_is_call_ongoing call_et.id + .then ([is_call_ongoing, response]) => + if not is_call_ongoing + event = @_fake_on_state_event response, call_et.id + @logger.log @logger.levels.DEBUG, "Call in '#{call_et.id}' ended during while connectivity was lost", event + @on_call_state event, true + + + ############################################################################### + # Call states + ############################################################################### + + ### + Handling of 'call.state' events. + @param event [Object] Event payload + @param client_joined_change [Boolean] Client joined state change triggered by client action + ### + on_call_state: (event, client_joined_change = false) -> + participant_ids = @_get_remote_participant_ids event.participants + self_user_joined = @_is_self_user_joined event.participants + participants_count = participant_ids.length + joined_count = @_get_joined_count participants_count, self_user_joined + + # Update existing call + @call_center.get_call_by_id event.conversation + .then (call_et) => + if event.self? and not call_et.self_client_joined() + if event.self.state is z.calling.enum.ParticipantState.JOINED or event.self.reason is 'ended' + client_joined_change = true + + event = @_should_ignore_state event, call_et, client_joined_change + return if not event + + if joined_count >= 1 + @_update_call event, participant_ids + @_update_self call_et, self_user_joined, client_joined_change + # ...which has ended + else + @delete_call call_et.id + .catch => + # Call with us joined + if self_user_joined + # ...from this device + if client_joined_change + @_create_outgoing_call event + # ...from another device + else + @_create_ongoing_call event, participant_ids + # ...with other participants + # New call we are not joined + else if participants_count > 0 + @_create_incoming_call event, participant_ids + + ### + Create the payload for to be set as call state. + + @private + @param state [z.calling.enum.ParticipantState] Self participant joined state + @return [Object] State object + ### + _create_state_payload: (state) -> + if state is z.calling.enum.ParticipantState.IDLE + self_state = + state: z.calling.enum.ParticipantState.IDLE + muted: false + screen_shared: false + videod: false + else + self_state = + state: z.calling.enum.ParticipantState.JOINED + muted: @self_state.muted() + screen_shared: @self_state.screen_shared() + videod: @self_state.videod() + + return self_state + + ### + Get the call state for a conversation. + @private + @param conversation_id [String] Conversation ID + ### + _get_state: (conversation_id) -> + if not conversation_id + error_description = 'GETting the call state not possible without conversation ID' + Raygun.send new Error error_description + @logger.log @logger.levels.ERROR, error_description + return Promise.reject new Error error_description + + @logger.log @logger.levels.INFO, "GETting call state for '#{conversation_id}'" + @call_center.call_service.get_state conversation_id, [] + .catch (error) => + @logger.log @logger.levels.ERROR, "GETting call state for '#{conversation_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'get', request: 'state'} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + throw error + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request conversation_id, jqXHR + @logger.log @logger.levels.DEBUG, "GETting call state for '#{conversation_id}' successful", response + return response + + ### + Put the clients call state for a conversation. + + @private + @param conversation_id [String] Conversation ID + @param payload [Object] Participant payload to be set + ### + _put_state: (conversation_id, payload) -> + if not conversation_id + error_description = "PUTting the state to '#{payload.state}' not possible without conversation ID" + @call_center.telemetry.report_error error_description + return Promise.reject new Error error_description + + @logger.log @logger.levels.INFO, + "PUTting the state to '#{payload.state}' in '#{conversation_id}'", payload + @call_center.call_service.put_state conversation_id, payload, [] + .catch (error) => + @_put_state_failure error, conversation_id, payload + .then (response_array) => + [response, jqXHR] = response_array + @call_center.telemetry.trace_request conversation_id, jqXHR + @logger.log @logger.levels.DEBUG, + "PUTting the state to '#{payload.state}' in '#{conversation_id}' successful", response + return response + + ### + Putting the clients call state for a conversation failed. + + @note Possible errors: + - {max_members: 25, member_count: 26, code: 409, message: "too many members for calling", label: "conv-too-big"} + - {"max_joined":9,"code":409,"message":"the voice channel is full","label":"voice-channel-full"} + - {"code":404,"message":"conversation not found","label":"not-found"} + + @private + @param error [JSON] Error object from the backend + @param conversation_id [String] Conversation ID + @param payload [Object] Participant payload to be set + ### + _put_state_failure: (error, conversation_id, payload) -> + @logger.log @logger.levels.ERROR, + "PUTting the state to '#{payload.state}' in '#{conversation_id}' failed: #{error.message}", error + attributes = {cause: error.label or error.name, method: 'put', request: 'state', video: payload.videod} + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUEST, undefined, attributes + @call_center.media_stream_handler.release_media_streams() + switch error.label + when z.service.BackendClientError::LABEL.CONVERSATION_TOO_BIG + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALL_FULL_CONVERSATION, + data: error.max_members + throw new z.calling.CallError error.message, z.calling.CallError::TYPE.CONVERSATION_TOO_BIG + when z.service.BackendClientError::LABEL.INVALID_OPERATION + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALL_EMPTY_CONVERSATION + @delete_call conversation_id + throw new z.calling.CallError error.message, z.calling.CallError::TYPE.CONVERSATION_EMPTY + when z.service.BackendClientError::LABEL.VOICE_CHANNEL_FULL + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALL_FULL_VOICE_CHANNEL, + data: error.max_joined + throw new z.calling.CallError error.message, z.calling.CallError::TYPE.VOICE_CHANNEL_FULL + # User has been removed from conversation, call should be deleted + else + @call_center.telemetry.report_error "PUTting the state to '#{payload.state}' failed: #{error.message}", error + @delete_call conversation_id + throw error + + ### + Put the clients call state for a conversation to z.calling.enum.ParticipantState.IDLE. + @private + @param conversation_id [String] Conversation ID + ### + _put_state_to_idle: (conversation_id) -> + @_put_state conversation_id, @_create_state_payload z.calling.enum.ParticipantState.IDLE + .then (response) => + @on_call_state @_fake_on_state_event(response, conversation_id), true + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to change state for call '#{conversation_id}': #{error.message}" + + ### + Put the clients call state for a conversation to z.calling.enum.ParticipantState.JOINED. + + @private + @param conversation_id [String] Conversation ID + @param self_state [Object] Self state to change into + @param client_joined_change [Boolean] Did the self client joined state change + ### + _put_state_to_join: (conversation_id, self_state, client_joined_change = false) -> + @_put_state conversation_id, self_state + .then (response) => + @call_center.timings().time_step z.telemetry.calling.CallSetupSteps.STATE_PUT + event = @_fake_on_state_event response, conversation_id + event.session = @_fake_session_id() if not event.session + + @call_center.telemetry.track_session conversation_id, event + @on_call_state event, client_joined_change + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to change state for call '#{conversation_id}': #{error.message}" + + ### + Check sequence number of event and decide if it will be processed. + + @private + @param event [Object] Event payload + @param call_et [z.calling.Call] Call entity + @param client_joined_change [Boolean] Client state change + @return [Object, undefined] Event or undefined if it should be ignored + ### + _should_ignore_state: (event, call_et, client_joined_change) -> + if event.sequence > call_et.event_sequence + @logger.log @logger.levels.INFO, "State processed: Sequence '#{event.sequence}' > '#{call_et.event_sequence}'" + event.is_sequential = event.sequence is call_et.event_sequence + 1 + call_et.event_sequence = event.sequence + else if client_joined_change + @logger.log @logger.levels.INFO, + "State processed: Contains self client change but '#{event.sequence}' <= '#{call_et.event_sequence}'" + call_et.event_sequence = event.sequence + else if event.sequence <= call_et.event_sequence + @logger.log @logger.levels.WARN, "State ignored: Sequence '#{event.sequence}' <= '#{call_et.event_sequence}'" + event = undefined + return event + + + ############################################################################### + # Call actions + ############################################################################### + + ### + Delete a call entity. + @param conversation_id [String] Conversation ID of call to be deleted + ### + delete_call: (conversation_id) => + @call_center.get_call_by_id conversation_id + .then (call_et) => + @logger.log @logger.levels.INFO, "Delete call in conversation '#{conversation_id}'" + # Reset call and delete it afterwards + call_et.state z.calling.enum.CallState.DELETED + call_et.reset_call() + @calls.remove call_et + @call_center.media_stream_handler.reset_media_streams() + .catch (error) => + @logger.log @logger.levels.WARN, "No call found in conversation '#{conversation_id}' to delete", error + + ### + User action to ignore incoming call. + @param conversation_id [String] Conversation ID of call to be joined + ### + ignore_call: (conversation_id) => + @call_center.get_call_by_id conversation_id + .then (call_et) => + call_et.ignore() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.INCOMING_CALL_MUTED + @logger.log @logger.levels.INFO, "Call in '#{conversation_id}' ignored" + @call_center.media_stream_handler.reset_media_streams() + .catch (error) => + @logger.log @logger.levels.WARN, "No call found in conversation '#{conversation_id}' to ignore", error + + ### + User action to join a call. + @param conversation_id [String] Conversation ID of call to be joined + @param is_videod [Boolean] Is this a video call + ### + join_call: (conversation_id, is_videod) => + @call_center.timings new z.telemetry.calling.CallSetupTimings conversation_id + + is_outgoing_call = false + + @call_center.get_call_by_id conversation_id + .catch -> + is_outgoing_call = true + .then => + if is_outgoing_call and not z.calling.CallCenter.supports_calling() + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.UNSUPPORTED_OUTGOING_CALL + else + @call_center.conversation_repository.get_conversation_by_id conversation_id, (conversation_et) => + @_already_joined_in_call conversation_id, is_videod, is_outgoing_call + .then -> + if is_outgoing_call + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.VOICE_CALL_INITIATED + media_action = if is_videod then 'audio_call' else 'video_call' + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.COMPLETED_MEDIA_ACTION, + action: media_action, conversation_type: if conversation_et.is_one2one() then 'one_to_one' else 'group' + return true + + ### + User action to leave a call. + @param conversation_id [String] Conversation ID of call to be joined + @param has_call_dropped [Boolean] Optional information whether the call has dropped + ### + leave_call: (conversation_id, has_call_dropped = false) => + @call_center.media_stream_handler.release_media_streams() + @call_center.get_call_by_id conversation_id + .then (call_et) => + if has_call_dropped + call_et.finished_reason = z.calling.enum.CallFinishedReason.CONNECTION_DROPPED + else + call_et.finished_reason = z.calling.enum.CallFinishedReason.SELF_USER + @_put_state_to_idle conversation_id + .catch (error) => + @logger.log @logger.levels.WARN, "No call found in conversation '#{conversation_id}' to leave", error + + ### + Leave a call we are joined immediately in case the browser window is closed. + @note Should only used by "window.onbeforeunload". + ### + leave_call_on_beforeunload: => + conversation_id = @_self_client_on_a_call() + @leave_call conversation_id if conversation_id + + ### + Remove a participant from a call if he was removed from the group. + @param conversation_id [String] Conversation ID of call that the user should be removed from + @param user_id [String] ID of user to be removed + ### + remove_participant: (conversation_id, user_id) => + @call_center.get_call_by_id conversation_id + .then (call_et) -> + if participant_et = call_et.get_participant_by_id user_id + call_et.delete_participant participant_et, false + .catch (error) => + @logger.log @logger.levels.WARN, "No call found in conversation '#{conversation_id}' to remove participant from", error + + ### + User action to toggle the audio muted state of a call. + @param conversation_id [String] Conversation ID of call + ### + toggle_audio: (conversation_id) => + @call_center.media_stream_handler.toggle_microphone_muted() + .then => + return @_put_state_to_join conversation_id, @_create_state_payload z.calling.enum.ParticipantState.JOINED if conversation_id + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to toggle video state: #{error.message}", error + + ### + User action to toggle the call state. + @param conversation_id [String] Conversation ID of call for which state will be toggled + @param is_videod [Boolean] Is this a video call + ### + toggle_joined: (conversation_id, is_videod) => + if @_self_participant_on_a_call() is conversation_id + @leave_call conversation_id if @_self_client_on_a_call() + else + @join_call conversation_id, is_videod + + ### + User action to toggle the screen sharing state of a call. + @param conversation_id [String] Conversation ID of call + ### + toggle_screen: (conversation_id) => + @call_center.media_stream_handler.toggle_screen_shared() + .then => + return @_put_state_to_join conversation_id, @_create_state_payload z.calling.enum.ParticipantState.JOINED if conversation_id + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to toggle screen sharing state: #{error.message}", error + + ### + User action to toggle the video state of a call. + @param conversation_id [String] Conversation ID of call + ### + toggle_video: (conversation_id) => + @call_center.media_stream_handler.toggle_camera_paused() + .then => + return @_put_state_to_join conversation_id, @_create_state_payload z.calling.enum.ParticipantState.JOINED if conversation_id + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to toggle audio state: #{error.message}", error + + ### + Check whether we are actively participating in a call. + + @private + @param new_call_id [String] Conversation ID of call about to be joined + @param is_videod [Boolean] Is video enabled for this new call + @param is_outgoing_call [Boolean] Is the new call outgoing + @return [Promise] Promise that resolves when the new call was joined + ### + _already_joined_in_call: (new_call_id, is_videod, is_outgoing_call) => + return new Promise (resolve) => + ongoing_call_id = @_self_participant_on_a_call() + if ongoing_call_id + @logger.log @logger.levels.WARN, 'You cannot start a second call while already participating in another one.' + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALL_START_ANOTHER, + action: => + @leave_call ongoing_call_id + window.setTimeout => + @_join_call new_call_id, is_videod + .then -> resolve() + , 1000 + close: -> + amplify.publish z.event.WebApp.CALL.STATE.IGNORE, new_call_id if not is_outgoing_call + data: is_outgoing_call + else + @_join_call new_call_id, is_videod + .then -> resolve() + + ### + Check whether a call is ongoing in a conversation. + + @private + @param conversation_id [String] Conversation ID + @return [Promise] Promise that resolves with an array whether a call was found and the current call state + ### + _is_call_ongoing: (conversation_id) => + @_get_state conversation_id + .then (response) => + for id, participant of response.participants when participant.state is z.calling.enum.ParticipantState.JOINED + @logger.log @logger.levels.DEBUG, "Detected ongoing call in '#{conversation_id}'" + return [true, response] + return [false, response] + .catch (error) => + @logger.log @logger.levels.WARN, "Detecting ongoing call in '#{conversation_id}' failed: #{error.message}", error + + ### + Join a call and get a MediaStream. + + @private + @param conversation_id [String] ID of conversation to call in + @param is_videod [Boolean] Is call a video call + ### + _join_call: (conversation_id, is_videod) -> + @call_center.get_call_by_id conversation_id + .catch (error) -> + throw error if error.type isnt z.calling.CallError::TYPE.CALL_NOT_FOUND + .then => + if @call_center.media_stream_handler.has_active_streams() + @logger.log @logger.levels.INFO, 'MediaStream has already been initialized', @call_center.media_stream_handler.local_media_streams + else + return @call_center.media_stream_handler.initiate_media_stream conversation_id, is_videod + .then => + return @_put_state_to_join conversation_id, @_create_state_payload(z.calling.enum.ParticipantState.JOINED), true + .catch (error) => + @logger.log @logger.levels.ERROR, "Joining call in '#{conversation_id}' failed: #{error.name}", error + + ### + Update a call with new state. + + @private + @param event [Object] 'call.state' event containing info to update call + @param joined_participant_ids [Array] User IDs of joined participants + ### + _update_call: (event, joined_participant_ids) -> + conversation_id = event.conversation + + @call_center.get_call_by_id conversation_id + .then (call_et) => + @call_center.user_repository.get_users_by_id joined_participant_ids, (participant_ets) -> + # This happens if we leave an ongoing call or if we accept a call on another device that we have ignored. + limit = event.is_sequential and event.cause is z.calling.enum.CallStateEventCause.REQUESTED + call_et.update_participants (new z.calling.entities.Participant user_et for user_et in participant_ets), limit + call_et.update_remote_state event.participants + .catch (error) => + @logger.log @logger.levels.WARN, "No call found in conversation '#{conversation_id}' to update", error + + ### + Update the self states of the call. + + @private + @param call_et [z.calling.Call] Call entity to update the self status off + @param user_joined_change [Boolean] Is the self user joined in the call + @param client_joined_change [Boolean] Was the state of the client changed + ### + _update_self: (call_et, self_user_joined, client_joined_change) -> + call_et.self_user_joined self_user_joined + if client_joined_change + call_et.self_client_joined self_user_joined + + if call_et.self_user_joined() and not call_et.self_client_joined() + call_et.state z.calling.enum.CallState.ONGOING + else if call_et.state() in z.calling.enum.CallState.OUTGOING + call_et.state z.calling.enum.CallState.CONNECTING if call_et.get_number_of_participants() > 0 + else if call_et.state() in z.calling.enum.CallStateGroups.CAN_CONNECT + if call_et.self_client_joined() and client_joined_change + call_et.state z.calling.enum.CallState.CONNECTING + else if call_et.state() is z.calling.enum.CallState.CONNECTING + call_et.state z.calling.enum.CallState.ONGOING if not call_et.self_client_joined() + + if call_et.is_remote_videod() and call_et.is_ongoing_on_another_client() + @call_center.media_stream_handler.release_media_streams() + + + ############################################################################### + # Call entity creation + ############################################################################### + + ### + Constructs a call entity. + + @private + @param event [Object] 'call.state' event containing info to create call + @return [z.calling.Call] Call entity + ### + _create_call: (event) -> + @call_center.get_call_by_id event.conversation + .then (call_et) => + @logger.log @logger.levels.WARN, "Call entity for '#{event.conversation}' already exists", call_et + .catch => + conversation_et = @call_center.conversation_repository.get_conversation_by_id event.conversation + call_et = new z.calling.entities.Call conversation_et, @call_center.user_repository.self(), @call_center.telemetry + call_et.local_audio_stream = @call_center.media_stream_handler.local_media_streams.audio + call_et.local_video_stream = @call_center.media_stream_handler.local_media_streams.video + call_et.session_id event.session or @_fake_session_id() + call_et.event_sequence = event.sequence + conversation_et.call call_et + @calls.push call_et + return call_et + + ### + Constructs an incoming call entity. + + @private + @param event [Object] 'call.state' event containing info to create call + @param remote_participant_ids [Array] User IDs of remote participants joined in call + @return [z.calling.Call] Call entity + ### + _create_incoming_call: (event, remote_participant_ids) -> + @_create_call event + .then (call_et) => + creator_id = @call_center.get_creator_id event + remote_participant_ids.push creator_id if creator_id not in remote_participant_ids + @call_center.user_repository.get_users_by_id remote_participant_ids, (remote_user_ets) => + call_et.set_creator @call_center.user_repository.get_user_by_id creator_id + participant_ets = (new z.calling.entities.Participant user_et for user_et in remote_user_ets) + call_et.update_participants participant_ets + call_et.update_remote_state event.participants + call_et.state z.calling.enum.CallState.INCOMING + @call_center.telemetry.track_event z.tracking.EventName.CALLING.RECEIVED_CALL, call_et + @call_center.media_stream_handler.initiate_media_stream call_et.id, true if call_et.is_remote_videod() + @logger.log @logger.levels.DEBUG, + "Incoming '#{call_et.remote_media_type()}' call to '#{call_et.conversation_et.display_name()}'", call_et + + ### + Constructs an ongoing call entity. + + @private + @param event [Object] 'call.state' event containing info to create call + @param remote_participant_ids [Array, undefined] User IDs of remote participants joined in call + @return [z.calling.Call] Call entity + ### + _create_ongoing_call: (event, remote_participant_ids = []) -> + @_create_call event + .then (call_et) => + call_et.state z.calling.enum.CallState.ONGOING + call_et.self_user_joined true + @call_center.user_repository.get_users_by_id remote_participant_ids, (remote_user_ets) => + participant_ets = (new z.calling.entities.Participant user_et for user_et in remote_user_ets) + call_et.update_participants participant_ets + call_et.update_remote_state event.participants + conversation_name = call_et.conversation_et.display_name() + @logger.log @logger.levels.DEBUG, + "Ongoing '#{call_et.remote_media_type()}' call to '#{conversation_name}' from another client", call_et + + ### + Constructs an outgoing call entity. + + @private + @param event [Object] 'call.state' event containing info to create call + @return [z.calling.Call] Call entity + ### + _create_outgoing_call: (event) -> + @_create_call event + .then (call_et) => + call_et.state z.calling.enum.CallState.OUTGOING + call_et.self_client_joined true + call_et.self_user_joined true + call_et.set_creator @call_center.user_repository.self() + @logger.log @logger.levels.DEBUG, + "Outgoing '#{@call_center.media_stream_handler.local_media_type()}' call to '#{call_et.conversation_et.display_name()}'", call_et + @call_center.telemetry.track_event z.tracking.EventName.CALLING.INITIATED_CALL, call_et + return call_et + + + ############################################################################### + # Helper functions + ############################################################################### + + ### + Create a fake 'call.state' event from a request response. + + @private + @param event [Object] Request response to be changed into 'call.state' event + @param conversation_id [String] Conversation ID for request response + ### + _fake_on_state_event: (event, conversation_id) -> + event.conversation = conversation_id + event.type = z.event.Backend.CALL.STATE + event.cause = z.calling.enum.CallStateEventCause.REQUESTED + return event + + ### + Create a fake session ID. + + @note Backend does not always provide a session ID, so we have to fake one + @private + @return [String] Random faked session ID + ### + _fake_session_id: -> + @logger.log @logger.levels.WARN, 'There is no session ID. We faked one.' + return "FAKE-#{z.util.create_random_uuid()}" + + ### + Get the count of joined users. + + @private + @param remote_participant_count [Number] Count of remote participants + @param is_self_user_joined [Boolean] Is the self user joined in the call + @return [Number] Number of users joined in the call + ### + _get_joined_count: (remote_participant_count, is_self_user_joined) -> + remote_participant_count++ if is_self_user_joined + return remote_participant_count + + ### + Get the IDs of remote participants. + + @private + @param participant_ets [Object] Object containing participants + @return [Array] Array user user IDs of joined, remote participants + ### + _get_remote_participant_ids: (participants) -> + participant_ids = [] + for id, participant of participants when participant.state is z.calling.enum.ParticipantState.JOINED + participant_ids.push id if id isnt @call_center.user_repository.self().id + return participant_ids + + ### + Check if self user is joined in call event. + + @private + @param participants [Object] JSON object containing call participants + @return [Boolean] Is the self user joined in the call + ### + _is_self_user_joined: (participants) -> + self = participants[@call_center.user_repository.self().id] + return self?.state is z.calling.enum.ParticipantState.JOINED + + + ### + Check if self client is participating in a call. + @private + @return [String, Boolean] Conversation ID of call or false + ### + _self_client_on_a_call: -> + return call_et.id for call_et in @calls() when call_et.self_client_joined() + + ### + Check if self participant is participating in a call. + @private + @return [String, Boolean] Conversation ID of call or false + ### + _self_participant_on_a_call: -> + return call_et.id for call_et in @calls() when call_et.self_user_joined() diff --git a/app/script/calling/handler/MediaDevicesHandler.coffee b/app/script/calling/handler/MediaDevicesHandler.coffee new file mode 100644 index 00000000000..5edf6a198e1 --- /dev/null +++ b/app/script/calling/handler/MediaDevicesHandler.coffee @@ -0,0 +1,255 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.handler ?= {} + +# MediaDevices handler +class z.calling.handler.MediaDevicesHandler + ### + Construct a new MediaDevices handler. + @param call_center [z.calling.CallCenter] Call center with references to all other handlers + ### + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.calling.handler.MediaDevicesHandler', z.config.LOGGER.OPTIONS + + @available_devices = + audio_input: ko.observableArray [] + audio_output: ko.observableArray [] + screen_input: ko.observableArray [] + video_input: ko.observableArray [] + + @current_device_id = + audio_input: ko.observable 'default' + audio_output: ko.observable 'default' + screen_input: ko.observable() + video_input: ko.observable() + + @current_device_index = + audio_input: ko.observable 0 + audio_output: ko.observable 0 + screen_input: ko.observable 0 + video_input: ko.observable 0 + + @has_camera = ko.pureComputed => return @available_devices.video_input().length > 0 + @has_microphone = ko.pureComputed => return @available_devices.audio_input().length > 0 + + @get_media_devices() + .then => + if @available_devices.video_input().length + default_device_index = @available_devices.video_input().length - 1 + @current_device_id.video_input @available_devices.video_input()[default_device_index].deviceId + @current_device_index.video_input default_device_index + @_subscribe_to_observables() + + @_subscribe_to_devices() + + # Subscribe to Knockout observables. + _subscribe_to_observables: => + @available_devices.audio_input.subscribe (media_devices) => + @_update_current_index_from_devices z.calling.enum.MediaDeviceType.AUDIO_INPUT, media_devices if media_devices + + @available_devices.audio_output.subscribe (media_devices) => + @_update_current_index_from_devices z.calling.enum.MediaDeviceType.AUDIO_OUTPUT, media_devices if media_devices + + @available_devices.screen_input.subscribe (media_devices) => + @_update_current_index_from_devices z.calling.enum.MediaDeviceType.SCREEN_INPUT, media_devices if media_devices + + @available_devices.video_input.subscribe (media_devices) => + @_update_current_index_from_devices z.calling.enum.MediaDeviceType.VIDEO_INPUT, media_devices if media_devices + + @current_device_id.audio_input.subscribe (media_device_id) => + if media_device_id and @call_center.joined_call() + @call_center.media_stream_handler.replace_input_source z.calling.enum.MediaType.AUDIO + @_update_current_index_from_id z.calling.enum.MediaDeviceType.AUDIO_INPUT, media_device_id + + @current_device_id.audio_output.subscribe (media_device_id) => + if media_device_id and @call_center.joined_call() + @call_center.media_element_handler.switch_media_element_output media_device_id + @_update_current_index_from_id z.calling.enum.MediaDeviceType.AUDIO_OUTPUT, media_device_id + + @current_device_id.screen_input.subscribe (media_device_id) => + if media_device_id and @call_center.joined_call() and @call_center.media_stream_handler.local_media_type() is z.calling.enum.MediaType.SCREEN + @call_center.media_stream_handler.replace_input_source z.calling.enum.MediaType.SCREEN + @_update_current_index_from_id z.calling.enum.MediaDeviceType.SCREEN_INPUT, media_device_id + + @current_device_id.video_input.subscribe (media_device_id) => + if media_device_id and @call_center.joined_call() and @call_center.media_stream_handler.local_media_type() is z.calling.enum.MediaType.VIDEO + @call_center.media_stream_handler.replace_input_source z.calling.enum.MediaType.VIDEO + @_update_current_index_from_id z.calling.enum.MediaDeviceType.VIDEO_INPUT, media_device_id + + # Subscribe to MediaDevices updates if available. + _subscribe_to_devices: => + if navigator.mediaDevices.ondevicechange? + navigator.mediaDevices.ondevicechange = => + @logger.log @logger.levels.INFO, 'List of available MediaDevices has changed' + @get_media_devices() + + ### + Update list of available MediaDevices. + @return [Promise] Promise that resolves with all MediaDevices when the list has been updated + ### + get_media_devices: => + navigator.mediaDevices.enumerateDevices() + .then (media_devices) => + if media_devices + @_remove_all_devices() + for media_device in media_devices + switch media_device.kind + when z.calling.enum.MediaDeviceType.AUDIO_INPUT + @available_devices.audio_input.push media_device + when z.calling.enum.MediaDeviceType.AUDIO_OUTPUT + @available_devices.audio_output.push media_device + when z.calling.enum.MediaDeviceType.VIDEO_INPUT + @available_devices.video_input.push media_device + + @logger.log @logger.levels.INFO, 'Updated MediaDevice list', media_devices + return media_devices + else + throw new z.calling.CallError 'No MediaDevices found', z.calling.CallError::TYPE.NO_DEVICES_FOUND + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to update MediaDevice list: #{error.message}", error + + ### + Update list of available Screens. + @return [Promise] Promise that resolves with all screen sources when the list has been updated + ### + get_screen_sources: -> + return new Promise (resolve, reject) => + options = + types: ['screen'] + thumbnailSize: + width: 312 + height: 176 + + window.desktopCapturer.getSources options, (error, screen_sources) => + if error + reject error + else + @logger.log @logger.levels.INFO, "Found '#{screen_sources.length}' possible sources for screen sharing on Electron", screen_sources + @available_devices.screen_input screen_sources + if screen_sources.length is 1 + @current_device_id.screen_input '' + @logger.log @logger.levels.INFO, "Selected '#{screen_sources[0].name}' for screen sharing", screen_sources[0] + @current_device_id.screen_input screen_sources[0].id + resolve screen_sources + + # Toggle between the available cameras. + toggle_next_camera: => + @get_media_devices() + .then => + [current_device, current_index] = @_get_current_device @available_devices.video_input(), @current_device_id.video_input() + next_device = @available_devices.video_input()[z.util.iterate_array_index(@available_devices.video_input(), @current_device_index.video_input()) or 0] + @current_device_id.video_input next_device.deviceId + @logger.log @logger.levels.INFO, "Switching the active camera from '#{current_device.label or current_device.deviceId}' to '#{next_device.label or next_device.deviceId}'" + + # Toggle between the available screens. + toggle_next_screen: => + @get_screen_sources() + .then => + [current_device, current_index] = @_get_current_device @available_devices.screen_input(), @current_device_id.screen_input() + next_device = @available_devices.screen_input()[z.util.iterate_array_index(@available_devices.screen_input(), @current_device_index.screen_input()) or 0] + @current_device_id.screen_input next_device.id + @logger.log @logger.levels.INFO, "Switching the active screen from '#{current_device.name or current_device.id}' to '#{next_device.name or next_device.id}'" + + ### + Check for availability of selected devices. + @param is_videod [Boolean] Also check for video devices + ### + update_current_devices: (is_videod) => + @get_media_devices() + .then => + _check_device = (media_type, device_type) => + device_type = @_type_conversion device_type + device_id_observable = @current_device_id["#{device_type}"] + media_devices = @available_devices["#{device_type}"]() + [media_device, media_device_index] = @_get_current_device media_devices, device_id_observable() + if not media_device.deviceId + if updated_device = @available_devices["#{device_type}"]()[0] + @logger.log @logger.levels.WARN, + "Current '#{media_type}' device '#{device_id_observable()}' not found and replaced by '#{updated_device.name}'", media_devices + device_id_observable updated_device.deviceId + else + @logger.log @logger.levels.WARN, "Current '#{media_type}' device '#{device_id_observable()}' not found and reset'", media_devices + device_id_observable '' + + _check_device z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaDeviceType.AUDIO_INPUT + _check_device z.calling.enum.MediaType.VIDEO, z.calling.enum.MediaDeviceType.VIDEO_INPUT if is_videod + + ### + Get the currently selected MediaDevice. + + @param media_devices [Array] Array of MediaDevices + @param current_device_id [String] ID of selected MediaDevice + @return [Array] Selected MediaDevice and its array index + ### + _get_current_device: (media_devices, current_device_id) -> + for media_device, index in media_devices when media_device.deviceId is current_device_id or media_device.id is current_device_id + return [media_device, index] + return [{}, 0] + + ### + Remove all known MediaDevices from the lists. + @private + ### + _remove_all_devices: -> + @available_devices.audio_input.removeAll() + @available_devices.audio_output.removeAll() + @available_devices.video_input.removeAll() + + ### + Add underscore to MediaDevice types. + @private + @param device_type [String] Device type string to update + @return [String] + ### + _type_conversion: (device_type) -> + device_type = device_type.replace('input', '_input').replace 'output', '_output' + + ### + Update the current index by searching for the current device. + + @private + @param index_observable [ko.obserable] Observable containing the current index + @param available_devices [Array] Array of MediaDevices + @param current_device_id [String] Current device ID to look for + ### + _update_current_device_index: (index_observable, available_devices, current_device_id) -> + [media_device, current_device_index] = @_get_current_device available_devices, current_device_id + index_observable current_device_index if _.isNumber current_device_index + + ### + Update the index for current device after the list of devices changed. + @private + @param device_type [z.calling.enum.MediaDeviceType] MediaDeviceType to be updates + @param available_devices [Array] Array of MediaDevices + ### + _update_current_index_from_devices: (device_type, available_devices) => + device_type = @_type_conversion device_type + @_update_current_device_index @current_device_index["#{device_type}"], available_devices, @current_device_id["#{device_type}"]() + + ### + Update the index for current device after the current device changed. + @private + @param device_type [z.calling.enum.MediaDeviceType] MediaDeviceType to be updates + @param selected_input_device_id [String] ID of selected input device + ### + _update_current_index_from_id: (device_type, selected_input_device_id) -> + device_type = @_type_conversion device_type + @_update_current_device_index @current_device_index["#{device_type}"], @available_devices["#{device_type}"](), selected_input_device_id diff --git a/app/script/calling/handler/MediaElementHandler.coffee b/app/script/calling/handler/MediaElementHandler.coffee new file mode 100644 index 00000000000..b7c4a9524e6 --- /dev/null +++ b/app/script/calling/handler/MediaElementHandler.coffee @@ -0,0 +1,109 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.handler ?= {} + +# MediaElement handler +class z.calling.handler.MediaElementHandler + ### + Construct a new MediaElement handler. + @param call_center [z.calling.CallCenter] Call center with references to all other handlers + ### + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.calling.handler.MediaElementHandler', z.config.LOGGER.OPTIONS + + @remote_media_elements = ko.observableArray [] + + ### + Add MediaElement for new stream. + @param media_stream_info [z.calling.payload.MediaStreamInfo] MediaStream information + ### + add_media_element: (media_stream_info) => + remote_media_element = @_create_media_element media_stream_info + @remote_media_elements.push remote_media_element + @logger.log @logger.levels.INFO, "Created MediaElement of type '#{remote_media_element.nodeName.toLowerCase()}' for MediaStream of flow '#{media_stream_info.flow_id}'", remote_media_element + + ### + Destroy the remote media element of a flow. + @private + @param flow_id [String] Flow ID for which to destroy the remote media element + ### + remove_media_element: (flow_id) => + for media_element in @_get_media_elements flow_id + @_destroy_media_element media_element + @remote_media_elements.remove media_element + @logger.log @logger.levels.INFO, "Deleted MediaElement of type '#{media_element.tagName.toLocaleLowerCase()}' for flow '#{flow_id}'" + + ### + Switch the output device used for all MediaElements. + @param media_device_id [String] Media Device ID to be used for playback + ### + switch_media_element_output: (media_device_id) => + @_set_media_element_output media_element, media_device_id for media_element in @remote_media_elements() + + ### + Create a new media element. + + @private + @param media_stream_info [z.calling.payload.MediaStreamInfo] MediaStream information + @return [HTMLMediaElement] HTMLMediaElement of type HTMLAudioElement that has the stream attached to it + ### + _create_media_element: (media_stream_info) -> + try + media_element = document.createElement 'audio' + media_element.srcObject = media_stream_info.stream + media_element.dataset['conversation_id'] = media_stream_info.conversation_id + media_element.dataset['flow_id'] = media_stream_info.flow_id + media_element.muted = false + media_element.setAttribute 'autoplay', true + return media_element + catch error + @logger.log @logger.levels.ERROR, + "Unable to create AudioElement for flow '#{media_stream_info.flow_id}'", error + + ### + Stop the media element. + @param media_element [HTMLMediaElement] A HTMLMediaElement that has the media stream attached to it + ### + _destroy_media_element: (media_element) -> + return if not media_element + media_element.pause() + media_element.srcObject = undefined + + ### + Get all the MediaElements related to a given flow ID. + @param flow_id [String] ID of flow to search MediaElements for + @return [Array] Related MediaElements + ### + _get_media_elements: (flow_id) -> + return (media_element for media_element in @remote_media_elements() when flow_id is media_element.dataset['flow_id']) + + ### + Change the output device used for audio playback of a media element. + @param media_element [HTMLMediaElement] HTMLMediaElement to change playback device for + @param sink_id [String] ID of MediaDevice to be used + ### + _set_media_element_output: (media_element, sink_id) -> + media_element.setSinkId sink_id + .then => + @logger.log @logger.levels.INFO, "Audio output device attached to flow '#{media_element.dataset['flow_id']} changed to '#{sink_id}'", media_element + .catch (error) => + @logger.log @logger.levels.INFO, + "Failed to attach audio output device '#{sink_id}' to flow '#{media_element.dataset['flow_id']}: #{error.message}", error diff --git a/app/script/calling/handler/MediaStreamHandler.coffee b/app/script/calling/handler/MediaStreamHandler.coffee new file mode 100644 index 00000000000..9f0623caf3e --- /dev/null +++ b/app/script/calling/handler/MediaStreamHandler.coffee @@ -0,0 +1,625 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.handler ?= {} + +# MediaStream handler +class z.calling.handler.MediaStreamHandler + ### + Detect whether a MediaStream has a video MediaStreamTrack attached + @param media_stream [MediaStream] MediaStream to detect the type off + @return [MediaStream] MediaStream with new type information + ### + @detect_media_stream_type: (media_stream) -> + if media_stream.getVideoTracks()?.length + if media_stream.getAudioTracks()?.length + media_stream.type = z.calling.enum.MediaType.AUDIO_VIDEO + else + media_stream.type = z.calling.enum.MediaType.VIDEO + else if media_stream.getAudioTracks()?.length + media_stream.type = z.calling.enum.MediaType.AUDIO + else + media_stream.type = z.calling.enum.MediaType.NONE + return media_stream + + ### + Get MediaStreamTracks from a MediaStream. + + @param media_stream [MediaStream] MediaStream to get tracks from + @param media_type [z.calling.enum.MediaType.AUDIO_VIDEO] + @return [Array] Array of MediaStreamTracks optionally matching the requested type + ### + @get_media_tracks: (media_stream, media_type = z.calling.enum.MediaType.AUDIO_VIDEO) -> + switch media_type + when z.calling.enum.MediaType.AUDIO + media_stream_tracks = media_stream.getAudioTracks() + when z.calling.enum.MediaType.AUDIO_VIDEO + media_stream_tracks = media_stream.getTracks() + when z.calling.enum.MediaType.VIDEO + media_stream_tracks = media_stream.getVideoTracks() + return media_stream_tracks + + ### + Construct a new MediaStream handler. + @param call_center [z.calling.CallCenter] Call center with references to all other handlers + ### + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.calling.handler.MediaDevicesHandler', z.config.LOGGER.OPTIONS + + @local_media_streams = + audio: ko.observable() + video: ko.observable() + + @remote_media_streams = + audio: ko.observableArray [] + video: ko.observable() + + @self_stream_state = + muted: ko.observable false + screen_shared: ko.observable false + videod: ko.observable false + + @local_media_type = ko.observable z.calling.enum.MediaType.AUDIO + + @has_active_streams = ko.pureComputed => + return @local_media_streams.audio()?.active or @local_media_streams.video()?.active + + @request_hint_timeout = undefined + + @local_media_streams.audio.subscribe (media_stream) => + if media_stream + @logger.log @logger.levels.DEBUG, "Local MediaStream contains MediaStreamTrack of kind 'audio'", + {stream: media_stream, audio_tracks: media_stream.getAudioTracks()} + @local_media_streams.video.subscribe (media_stream) => + if media_stream + @logger.log @logger.levels.DEBUG, "Local MediaStream contains MediaStreamTrack of kind 'video'", + {stream: media_stream, video_tracks: media_stream.getVideoTracks()} + + @current_device_id = @call_center.media_devices_handler.current_device_id + + amplify.subscribe z.event.WebApp.CALL.MEDIA.ADD_STREAM, @add_remote_media_stream + + + ############################################################################### + # MediaStream constraints + ############################################################################### + + ### + Get the MediaStreamConstraints to be used for MediaStream creation. + + @private + @param request_audio [Boolean] Request audio in the constraints + @param request_video [Boolean] Request video in the constraints + @return [Object] MediaStreamConstraints + ### + get_media_stream_constraints: (request_audio = false, request_video = false) -> + Promise.resolve() + .then => + constraints = + audio: if request_audio then @_get_audio_stream_constraints @current_device_id.audio_input() else undefined + video: if request_video then @_get_video_stream_constraints @current_device_id.video_input() else undefined + @logger.log @logger.levels.INFO, 'Set constraints for MediaStream', constraints + media_type = if request_video then z.calling.enum.MediaType.VIDEO else z.calling.enum.MediaType.AUDIO + return [media_type, constraints] + + ### + Get the MediaStreamConstraints to be used for screen sharing. + @return [Object] MediaStreamConstraints + ### + get_screen_stream_constraints: => + return new Promise (resolve, reject) => + if window.desktopCapturer + @logger.log @logger.levels.INFO, 'Enabling screen sharing from Electron' + + constraints = + audio: false + video: + mandatory: + chromeMediaSource: 'desktop' + chromeMediaSourceId: @current_device_id.screen_input() + maxHeight: 720 + maxWidth: 1280 + minWidth: 1280 + minHeight: 720 + + resolve [z.calling.enum.MediaType.SCREEN, constraints] + + else if z.util.Environment.browser.firefox + @logger.log @logger.levels.INFO, 'Enabling screen sharing from Firefox' + + constraints = + audio: false + video: + mediaSource: 'screen' + + resolve [z.calling.enum.MediaType.SCREEN, constraints] + else + reject new z.calling.CallError 'Screen sharing is not yet supported by this browser', z.calling.CallError::TYPE.NOT_SUPPORTED + + ### + Get the video constraints to be used for MediaStream creation. + @private + @param media_device_id [String] Optional ID of MediaDevice to be used + @return [Object] Video stream constraints + ### + _get_audio_stream_constraints: (media_device_id) -> + if _.isString media_device_id and media_device_id isnt 'default' + media_stream_constraints = + deviceId: + exact: media_device_id + else + media_stream_constraints = true + + return media_stream_constraints + + ### + Get the video constraints to be used for MediaStream creation. + @private + @param media_device_id [String] Optional ID of MediaDevice to be used + @return [Object] Video stream constraints + ### + _get_video_stream_constraints: (media_device_id) -> + media_stream_constraints = + facingMode: 'user' + frameRate: 30 + width: + min: 640 + ideal: 640 + max: 1280 + height: + min: 360 + ideal: 360 + max: 720 + + if _.isString media_device_id + media_stream_constraints.deviceId = + exact: media_device_id + + return media_stream_constraints + + + ############################################################################### + # Local MediaStream handling + ############################################################################### + + ### + Initiate the MediaStream. + @param conversation_id [String] Conversation ID of call + @param is_videod [Boolean] Should MediaStreamContain video + @return [Promise] Promise that resolve when the MediaStream has been initiated + ### + initiate_media_stream: (conversation_id, is_videod) => + @call_center.media_devices_handler.update_current_devices is_videod + .then => + return @get_media_stream_constraints true, is_videod + .then ([media_type, media_stream_constraints]) => + return @request_media_stream media_type, media_stream_constraints, conversation_id + .then (media_stream_info) => + @self_stream_state.videod is_videod + @local_media_type z.calling.enum.MediaType.VIDEO if is_videod + @_initiate_media_stream_success media_stream_info + .catch (error) => + if _.isArray error + [error, media_type] = error + @_initiate_media_stream_failure error, media_type, conversation_id + @logger.log @logger.levels.ERROR, "Requesting MediaStream failed: #{error.name}", error + @call_center.telemetry.track_event z.tracking.EventName.CALLING.FAILED_REQUESTING_MEDIA, undefined, {cause: error.name, video: is_videod} + + # Release the MediaStreams. + release_media_streams: => + media_streams_identical = @_compare_local_media_streams() + + @local_media_streams.audio undefined if @_release_media_stream @local_media_streams.audio() + @local_media_streams.video undefined if media_streams_identical or @_release_media_stream @local_media_streams.video() + + ### + Replace the MediaStream after a change of the selected input device. + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Info about new MediaStream + ### + replace_media_stream: (media_stream_info) => + @logger.log @logger.levels.DEBUG, "Received new MediaStream with '#{media_stream_info.stream.getTracks().length}' MediaStreamTrack/s", + {stream: media_stream_info.stream, audio_tracks: media_stream_info.stream.getAudioTracks(), video_tracks: media_stream_info.stream.getVideoTracks()} + @_set_stream_state media_stream_info + return Promise.all (flow_et.switch_media_stream media_stream_info for flow_et in @call_center.joined_call().get_flows()) + .then (resolve_array) => + [media_stream_info, replaced_stream] = resolve_array[0] + if replaced_stream + @release_media_streams media_stream_info.type + else + if media_stream_info.type is z.calling.enum.MediaType.VIDEO + @_release_media_stream @local_media_streams.video(), z.calling.enum.MediaType.VIDEO + else + @_release_media_stream @local_media_streams.audio(), z.calling.enum.MediaType.AUDIO + @set_local_media_stream media_stream_info + + ### + Update the used MediaStream after a new input device was selected. + @param type [z.calling.enum.MediaType] Media type of device that was replaced + ### + replace_input_source: (media_type) => + return if not @needs_media_stream() + + switch media_type + when z.calling.enum.MediaType.AUDIO + constraints_promise = @get_media_stream_constraints true, false + when z.calling.enum.MediaType.SCREEN + constraints_promise = @get_screen_stream_constraints() + when z.calling.enum.MediaType.VIDEO + request_audio = not z.util.Environment.browser.firefox + constraints_promise = @get_media_stream_constraints request_audio, true + + constraints_promise.then ([media_type, media_stream_constraints]) => + return @request_media_stream media_type, media_stream_constraints + .then (media_stream_info) => + @_set_self_stream_state media_type + return @replace_media_stream media_stream_info + .catch (error) => + [error, media_type] = error if _.isArray error + @_replace_input_source_failure error, media_type + + ### + Request a MediaStream. + + @param media_type [z.calling.enum.MediaType] Type of MediaStream to be requested + @param media_stream_constraints [RTCMediaStreamConstraints] Constraints for the MediaStream to be requested + @param conversation_id [String] Conversation ID + @return [Promise] Promise that will resolve with an array of the stream type and the stream + ### + request_media_stream: (media_type, media_stream_constraints, conversation_id) => + return new Promise (resolve, reject) => + if not @call_center.media_devices_handler.has_microphone() + @logger.log @logger.levels.WARN, "Requesting MediaStream access aborted - 'No microphone'" + @_show_device_not_found_hint z.calling.enum.MediaType.AUDIO, conversation_id + reject new z.calling.CallError 'No microphone found', z.calling.CallError::TYPE.NO_MICROPHONE_FOUND + else if not @call_center.media_devices_handler.has_camera() and media_stream_constraints.video + @logger.log @logger.levels.WARN, "Requesting MediaStream access aborted - 'No camera'" + @_show_device_not_found_hint z.calling.enum.MediaType.VIDEO, conversation_id + reject new z.calling.CallError 'No camera found', z.calling.CallError::TYPE.NO_CAMERA_FOUND + else + @logger.log @logger.levels.INFO, "Requesting MediaStream access for '#{media_type}'", media_stream_constraints + @request_hint_timeout = window.setTimeout => + @_hide_permission_failed_hint media_type + @_show_permission_request_hint media_type + @request_hint_timeout = undefined + , 200 + + @call_center.timings().time_step z.telemetry.calling.CallSetupSteps.STREAM_REQUESTED if @call_center.timings() + navigator.mediaDevices.getUserMedia media_stream_constraints + .then (media_stream) => + @_clear_permission_request_hint media_type + resolve new z.calling.payloads.MediaStreamInfo z.calling.enum.MediaStreamSource.LOCAL, 'self', media_stream + .catch (error) => + @_clear_permission_request_hint media_type + reject [error, media_type] + + ### + Save a reference to a local MediaStream. + @param media_stream_info [z.calling.payloads.MediaStreamInfo] MediaStream and meta information + ### + set_local_media_stream: (media_stream_info) => + if media_stream_info.type in [z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaType.AUDIO_VIDEO] + @local_media_streams.audio media_stream_info.stream + if media_stream_info.type in [z.calling.enum.MediaType.AUDIO_VIDEO, z.calling.enum.MediaType.VIDEO] + @local_media_streams.video media_stream_info.stream + + ### + Clear the permission request hint timeout or hide the warning. + @private + @param media_type [z.calling.enum.MediaType] Type of requested stream + ### + _clear_permission_request_hint: (media_type) -> + if @request_hint_timeout + window.clearTimeout @request_hint_timeout + else + @_hide_permission_request_hint media_type + + ### + Compare the local MediaStreams for equality. + @private + @return [Boolean] True if both audio and video stream are identical + ### + _compare_local_media_streams: -> + return @local_media_streams.audio()?.id is @local_media_streams.video()?.id + + ### + Hide the permission denied hint banner. + @private + @param media_type [z.calling.enum.MediaType] Type of requested stream + ### + _hide_permission_failed_hint: (media_type) -> + switch media_type + when z.calling.enum.MediaType.AUDIO + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.DENIED_MICROPHONE + when z.calling.enum.MediaType.SCREEN + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.DENIED_SCREEN + when z.calling.enum.MediaType.VIDEO + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.DENIED_CAMERA + + ### + Hide the permission request hint banner. + @private + @param media_type [z.calling.enum.MediaType] Type of requested stream + ### + _hide_permission_request_hint: (media_type) -> + return if z.util.Environment.electron + switch media_type + when z.calling.enum.MediaType.AUDIO + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.REQUEST_MICROPHONE + when z.calling.enum.MediaType.SCREEN + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.REQUEST_SCREEN + when z.calling.enum.MediaType.VIDEO + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.REQUEST_CAMERA + + ### + Initial request for local MediaStream was successful. + @private + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Type of requested MediaStream + ### + _initiate_media_stream_success: (media_stream_info) => + return if not media_stream_info + @call_center.timings().time_step z.telemetry.calling.CallSetupSteps.STREAM_RECEIVED if @call_center.timings() + @logger.log @logger.levels.DEBUG, "Received initial MediaStream with '#{media_stream_info.stream.getTracks().length}' MediaStreamTrack/s", + {stream: media_stream_info.stream, audio_tracks: media_stream_info.stream.getAudioTracks(), video_tracks: media_stream_info.stream.getVideoTracks()} + @set_local_media_stream media_stream_info + + ### + Local MediaStream creation failed. + + @private + @param error [MediaStreamError] Error message from navigator.MediaDevices.getUserMedia() + @param media_type [z.calling.enum.MediaType] Type of requested MediaStream + @param conversation_id [String] Conversation ID + ### + _initiate_media_stream_failure: (error, media_type, conversation_id) => + if error.name in z.calling.rtc.MediaStreamErrorTypes.PERMISSION + @_show_permission_denied_hint media_type + else if error.name in z.calling.rtc.MediaStreamErrorTypes.MISC + @_show_permission_denied_hint media_type + else if error.name in z.calling.rtc.MediaStreamErrorTypes.DEVICE + @_show_device_not_found_hint media_type, conversation_id + + ### + Release the MediaStream. + + @private + @param stream [MediaStream] MediaStream to be released + @param media_type [z.calling.enum.MediaType] Type of MediaStreamTracks to be released + ### + _release_media_stream: (media_stream, media_type = z.calling.enum.MediaType.AUDIO_VIDEO) => + return false if not media_stream + + media_stream_tracks = z.calling.handler.MediaStreamHandler.get_media_tracks media_stream, media_type + + if media_stream_tracks.length + for media_stream_track in media_stream_tracks + media_stream.removeTrack media_stream_track + media_stream_track.stop() + @logger.log @logger.levels.INFO, "Stopping MediaStreamTrack of kind '#{media_stream_track.kind}' successful", media_stream_track + return true + else + @logger.log @logger.levels.WARN, 'No MediaStreamTrack found to stop', media_stream + return false + + ### + Failed to replace an input source. + + @private + @param error [Error] Error thrown when attempting to replace the source + @param media_type [z.calling.enum.MediaType] Type of failed request + ### + _replace_input_source_failure: (error, media_type) -> + if media_type is z.calling.enum.MediaType.SCREEN + if z.util.Environment.browser.firefox and error.name is z.calling.rtc.MediaStreamError.NOT_ALLOWED_ERROR + @logger.log @logger.levels.WARN, 'We are not on the white list. Manually add the current domain to media.getusermedia.screensharing.allowed_domains on about:config' + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.WHITELIST_SCREENSHARING + else + @logger.log @logger.levels.ERROR, "Failed to enable screen sharing: #{error.message}", error + else + @logger.log @logger.levels.ERROR, "Failed to replace '#{media_type}' input source: #{error.message}", error + + ### + Show microphone not found hin banner. + + @private + @param media_type [z.calling.enum.MediaType] Type of device not found + @param conversation_id [String] Optional conversation ID + ### + _show_device_not_found_hint: (media_type, conversation_id) -> + if media_type is z.calling.enum.MediaType.AUDIO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.NOT_FOUND_MICROPHONE + else if media_type is z.calling.enum.MediaType.VIDEO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.NOT_FOUND_CAMERA + amplify.publish z.event.WebApp.CALL.STATE.IGNORE, conversation_id if conversation_id + + ### + Show permission denied hint banner. + @private + @param media_type [z.calling.enum.MediaType] Type of media access request + ### + _show_permission_denied_hint: (media_type) -> + switch media_type + when z.calling.enum.MediaType.AUDIO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.DENIED_MICROPHONE + when z.calling.enum.MediaType.SCREEN + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.DENIED_SCREEN + when z.calling.enum.MediaType.VIDEO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.DENIED_CAMERA + + ### + Show permission request hint banner. + @private + @param media_type [z.calling.enum.MediaType] Type of requested MediaStream + ### + _show_permission_request_hint: (media_type) -> + return if z.util.Environment.electron + switch media_type + when z.calling.enum.MediaType.AUDIO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.REQUEST_MICROPHONE + when z.calling.enum.MediaType.SCREEN + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.REQUEST_SCREEN + when z.calling.enum.MediaType.VIDEO + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.REQUEST_CAMERA + + + ############################################################################### + # Remote MediaStream handling + ############################################################################### + + ### + Add a remote MediaStream. + @param media_stream_info [z.calling.payload.MediaStreamInfo] MediaStream information + ### + add_remote_media_stream: (media_stream_info) => + switch media_stream_info.type + when z.calling.enum.MediaType.AUDIO + @remote_media_streams.audio.push media_stream_info.stream + @call_center.media_element_handler.add_media_element media_stream_info + when z.calling.enum.MediaType.AUDIO_VIDEO, z.calling.enum.MediaType.VIDEO + @remote_media_streams.video media_stream_info.stream + + + ############################################################################### + # Media handling + ############################################################################### + + ### + Check for active calls that need a MediaStream. + @return [Boolean] Returns true if an active media stream is needed for at least one call + ### + needs_media_stream: -> + for call_et in @call_center.calls() + return true if call_et.is_remote_videod() and call_et.state() is z.calling.enum.CallState.INCOMING + return true if call_et.self_client_joined() + return false + + # Toggle the camera. + toggle_camera_paused: => + if @local_media_streams.video() and @local_media_type() is z.calling.enum.MediaType.VIDEO + @_toggle_video_enabled() + else + @replace_input_source z.calling.enum.MediaType.VIDEO + + # Toggle the mute state of the microphone. + toggle_microphone_muted: => + @_toggle_audio_enabled() if @local_media_streams.audio() + + # Toggle the screen. + toggle_screen_shared: => + if @local_media_streams.video() and @local_media_type() is z.calling.enum.MediaType.SCREEN + @_toggle_screen_enabled() + else + @replace_input_source z.calling.enum.MediaType.SCREEN + + # Reset the enabled states of media types. + reset_self_states: => + @self_stream_state.muted false + @self_stream_state.screen_shared false + @self_stream_state.videod false + @local_media_type z.calling.enum.MediaType.AUDIO + + # Reset the MediaStreams and states. + reset_media_streams: => + if not @needs_media_stream() + @call_center.audio_repository.close_audio_context() + @release_media_streams() + @reset_self_states() + + ### + Set the self stream state to reflect current call type. + @param media_type [z.calling.enum.MediaType] Type of state to enable + ### + _set_self_stream_state: (media_type) -> + switch media_type + when z.calling.enum.MediaType.AUDIO + @self_stream_state.muted false + when z.calling.enum.MediaType.SCREEN + @self_stream_state.videod false + @self_stream_state.screen_shared true + @local_media_type z.calling.enum.MediaType.SCREEN + when z.calling.enum.MediaType.VIDEO + @self_stream_state.videod true + @self_stream_state.screen_shared false + @local_media_type z.calling.enum.MediaType.VIDEO + + ### + Set the enabled state of a new MediaStream. + @private + @param media_stream_info [z.calling.payloads.MediaStreamInfo] Info about MediaStream to set state off + ### + _set_stream_state: (media_stream_info) -> + if media_stream_info.type in [z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaType.AUDIO_VIDEO] + audio_stream_tracks = z.calling.handler.MediaStreamHandler.get_media_tracks media_stream_info.stream, z.calling.enum.MediaType.AUDIO + audio_stream_tracks[0].enabled = not @self_stream_state.muted() + + if media_stream_info.type in [z.calling.enum.MediaType.AUDIO_VIDEO, z.calling.enum.MediaType.VIDEO] + video_stream_tracks = z.calling.handler.MediaStreamHandler.get_media_tracks media_stream_info.stream, z.calling.enum.MediaType.VIDEO + video_stream_tracks[0].enabled = @self_stream_state.screen_shared() or @self_stream_state.videod() + + ### + Toggle the audio stream. + @private + ### + _toggle_audio_enabled: -> + @_toggle_stream_enabled z.calling.enum.MediaType.AUDIO, @local_media_streams.audio(), @self_stream_state.muted + .then (audio_track) => + @logger.log @logger.levels.INFO, "Microphone muted: #{@self_stream_state.muted()}", audio_track + return @self_stream_state.muted() + + ### + Toggle the screen stream. + @private + ### + _toggle_screen_enabled: -> + @_toggle_stream_enabled z.calling.enum.MediaType.VIDEO, @local_media_streams.video(), @self_stream_state.screen_shared + .then (video_track) => + @logger.log @logger.levels.INFO, "Screen enabled: #{@self_stream_state.screen_shared()}", video_track + return @self_stream_state.screen_shared() + + ### + Toggle the video stream. + @private + ### + _toggle_video_enabled: -> + @_toggle_stream_enabled z.calling.enum.MediaType.VIDEO, @local_media_streams.video(), @self_stream_state.videod + .then (video_track) => + @logger.log @logger.levels.INFO, "Camera enabled: #{@self_stream_state.videod()}", video_track + return @self_stream_state.videod() + + ### + Toggle the enabled state of a MediaStream. + + @private + @param media_type [z.calling.enum.MediaType] Media type to toggle + @param media_stream [MediaStream] MediaStream to toggle enabled state off + @param state_observable [ko.observable] State observable to invert + @return [MediaStreamTrack] Updated MediaStreamTrack with new enabled state + ### + _toggle_stream_enabled: (media_type, media_stream, state_observable) -> + Promise.resolve() + .then -> + state_observable not state_observable() + media_stream_track = (z.calling.handler.MediaStreamHandler.get_media_tracks media_stream, media_type)[0] + if media_type is z.calling.enum.MediaType.AUDIO + enabled_state = not state_observable() + amplify.publish z.event.WebApp.CALL.MEDIA.MUTE_AUDIO, state_observable() + else + enabled_state = state_observable() + media_stream_track.enabled = enabled_state + return media_stream_track diff --git a/app/script/calling/mapper/ICECandidateMapper.coffee b/app/script/calling/mapper/ICECandidateMapper.coffee new file mode 100644 index 00000000000..4e710c5c5d7 --- /dev/null +++ b/app/script/calling/mapper/ICECandidateMapper.coffee @@ -0,0 +1,45 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.mapper ?= {} + +class z.calling.mapper.ICECandidateMapper + constructor: -> + @logger = new z.util.Logger 'z.calling.mapper.ICECandidateMapper', z.config.LOGGER.OPTIONS + + ### + @param ice_candidate [RTCIceCandidate] Interactive Connectivity Establishment (ICE) Candidate + ### + map_ice_object_to_message: (ice_candidate) -> + message = + sdp: ice_candidate.candidate + sdp_mline_index: ice_candidate.sdpMLineIndex + sdp_mid: ice_candidate.sdpMid + + return message + + # We have to convert camel-case to underscores + map_ice_message_to_object: (ice_message) -> + candidate_info = + candidate: ice_message.sdp + sdpMLineIndex: ice_message.sdp_mline_index + sdpMid: ice_message.sdp_mid + + return new RTCIceCandidate candidate_info diff --git a/app/script/calling/mapper/SDPMapper.coffee b/app/script/calling/mapper/SDPMapper.coffee new file mode 100644 index 00000000000..72f12a906d2 --- /dev/null +++ b/app/script/calling/mapper/SDPMapper.coffee @@ -0,0 +1,41 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.mapper ?= {} + +class z.calling.mapper.SDPMapper + constructor: -> + @logger = new z.util.Logger 'z.calling.mapper.SDPMapper', z.config.LOGGER.OPTIONS + + map_sdp_event_to_object: (event) => + remote_sdp = + sdp: @_convert_sdp_fingerprint_to_uppercase event.sdp + type: event.state + + return new window.RTCSessionDescription remote_sdp + + _convert_sdp_fingerprint_to_uppercase: (sdp_string) -> + sdp_lines = sdp_string.split '\r\n' + + for sdp_line, index in sdp_lines when sdp_line.startsWith 'a=fingerprint' + sdp_line_parts = sdp_line.split ' ' + sdp_lines[index] = "#{sdp_line_parts[0]} #{sdp_line_parts[1].toUpperCase()}" + + return sdp_lines.join '\r\n' diff --git a/app/script/calling/payloads/FlowDeletionInfo.coffee b/app/script/calling/payloads/FlowDeletionInfo.coffee new file mode 100644 index 00000000000..0628f12ed94 --- /dev/null +++ b/app/script/calling/payloads/FlowDeletionInfo.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.payloads ?= {} + +z.calling.payloads.FlowDeletionReason = + RELEASED: 'released' + TIMEOUT: 'timeout' + +class z.calling.payloads.FlowDeletionInfo + constructor: (@conversation_id, @flow_id, @reason) -> + return @ diff --git a/app/script/calling/payloads/ICECandidateInfo.coffee b/app/script/calling/payloads/ICECandidateInfo.coffee new file mode 100644 index 00000000000..d00e3816403 --- /dev/null +++ b/app/script/calling/payloads/ICECandidateInfo.coffee @@ -0,0 +1,34 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.payloads ?= {} + +class z.calling.payloads.ICECandidateInfo + ### + Object to keep an ICE candidate bundled with signaling information. + + @param conversation_id [String] Conversation ID + @param flow_id [String] Flow ID + @param ice_candidate [RTCIceCandidate] Interactive Connectivity Establishment (ICE) Candidate + ### + constructor: (conversation_id, flow_id, ice_candidate) -> + @conversation_id = conversation_id + @flow_id = flow_id + @ice_candidate = ice_candidate diff --git a/app/script/calling/payloads/MediaStreamInfo.coffee b/app/script/calling/payloads/MediaStreamInfo.coffee new file mode 100644 index 00000000000..aecf513d4f5 --- /dev/null +++ b/app/script/calling/payloads/MediaStreamInfo.coffee @@ -0,0 +1,33 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.payloads ?= {} + +class z.calling.payloads.MediaStreamInfo + constructor: (@source, @flow_id, @stream, @call_et) -> + @type = z.calling.enum.MediaType.NONE + + @conversation_id = @call_et?.id + @update_stream_type() + return @ + + update_stream_type: => + @stream = z.calling.handler.MediaStreamHandler.detect_media_stream_type @stream + @type = @stream.type diff --git a/app/script/calling/payloads/SDPInfo.coffee b/app/script/calling/payloads/SDPInfo.coffee new file mode 100644 index 00000000000..f925521db3a --- /dev/null +++ b/app/script/calling/payloads/SDPInfo.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.payloads ?= {} + +class z.calling.payloads.SDPInfo + ### + Object to keep an SDP bundled with signaling information. + + @param params [Object] Properties to setup the ICE information container + @option params [String] conversation_id Conversation ID + @option params [String] flow_id Flow ID + @option params [RTCSessionDescription, mozRTCSessionDescription] sdp Session Description Protocol (SDP) + ### + constructor: (params) -> + @conversation_id = params.conversation_id + @flow_id = params.flow_id + @sdp = params.sdp diff --git a/app/script/calling/rtc/ICEConnectionState.coffee b/app/script/calling/rtc/ICEConnectionState.coffee new file mode 100644 index 00000000000..78d2d16ec2a --- /dev/null +++ b/app/script/calling/rtc/ICEConnectionState.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# http://www.w3.org/TR/webrtc/#rtciceconnectionstate-enum +# https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection.iceConnectionState#Value +z.calling.rtc.ICEConnectionState = + NEW: 'new' + CHECKING: 'checking' + CONNECTED: 'connected' + COMPLETED: 'completed' + FAILED: 'failed' + DISCONNECTED: 'disconnected' + CLOSED: 'closed' diff --git a/app/script/calling/rtc/ICEGatheringState.coffee b/app/script/calling/rtc/ICEGatheringState.coffee new file mode 100644 index 00000000000..3f82395c456 --- /dev/null +++ b/app/script/calling/rtc/ICEGatheringState.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# http://www.w3.org/TR/webrtc/#rtcicegatheringstate-enum +# https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection.iceGatheringState#Value +z.calling.rtc.ICEGatheringState = + NEW: 'new' + GATHERING: 'gathering' + COMPLETE: 'complete' diff --git a/app/script/calling/rtc/MediaStreamError.coffee b/app/script/calling/rtc/MediaStreamError.coffee new file mode 100644 index 00000000000..72e6f1f0d7c --- /dev/null +++ b/app/script/calling/rtc/MediaStreamError.coffee @@ -0,0 +1,37 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia#Errors +z.calling.rtc.MediaStreamError = + ABORT_ERROR: 'AbortError' + DEVICES_NOT_FOUND_ERROR: 'DevicesNotFoundError' + INTERNAL_ERROR: 'InternalError' + INVALID_STATE_ERROR: 'InvalidStateError' + NOT_ALLOWED_ERROR: 'NotAllowedError' + NOT_FOUND_ERROR: 'NotFoundError' + NOT_READABLE_ERROR: 'NotReadableError' + OVER_CONSTRAINED_ERROR: 'OverConstrainedError' + PERMISSION_DENIED_ERROR: 'PermissionDeniedError' + PERMISSION_DISMISSED_ERROR: 'PermissionDismissedError' + SECURITY_ERROR: 'SecurityError' + TYPE_ERROR: 'TypeError' + SOURCE_UNAVAILABLE_ERROR: 'SourceUnavailableError' diff --git a/app/script/calling/rtc/MediaStreamErrorTypes.coffee b/app/script/calling/rtc/MediaStreamErrorTypes.coffee new file mode 100644 index 00000000000..b7764dd91bc --- /dev/null +++ b/app/script/calling/rtc/MediaStreamErrorTypes.coffee @@ -0,0 +1,42 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia#Errors +z.calling.rtc.MediaStreamErrorTypes = + DEVICE: [ + z.calling.rtc.MediaStreamError.ABORT_ERROR + z.calling.rtc.MediaStreamError.DEVICES_NOT_FOUND_ERROR + z.calling.rtc.MediaStreamError.NOT_FOUND_ERROR + z.calling.rtc.MediaStreamError.NOT_READABLE_ERROR + ] + MISC: [ + z.calling.rtc.MediaStreamError.INTERNAL_ERROR + z.calling.rtc.MediaStreamError.INVALID_STATE_ERROR + z.calling.rtc.MediaStreamError.SOURCE_UNAVAILABLE_ERROR + z.calling.rtc.MediaStreamError.OVER_CONSTRAINED_ERROR + z.calling.rtc.MediaStreamError.TYPE_ERROR + ] + PERMISSION: [ + z.calling.rtc.MediaStreamError.PERMISSION_DENIED_ERROR + z.calling.rtc.MediaStreamError.PERMISSION_DISMISSED_ERROR + z.calling.rtc.MediaStreamError.SECURITY_ERROR + ] diff --git a/app/script/calling/rtc/SDPType.coffee b/app/script/calling/rtc/SDPType.coffee new file mode 100644 index 00000000000..fdcb9a66077 --- /dev/null +++ b/app/script/calling/rtc/SDPType.coffee @@ -0,0 +1,31 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# http://www.w3.org/TR/webrtc/#rtcsdptype +# https://developer.mozilla.org/en-US/docs/Web/API/RTCSessionDescription#RTCSdpType +z.calling.rtc.SDPType = + ANSWER: 'answer' + LOCAL: 'local' + REMOTE: 'remote' + OFFER: 'offer' + PROVISIONAL_ANSWER: 'pranswer' + ROLLBACK: 'rollback' diff --git a/app/script/calling/rtc/SignalingState.coffee b/app/script/calling/rtc/SignalingState.coffee new file mode 100644 index 00000000000..35d75a4741b --- /dev/null +++ b/app/script/calling/rtc/SignalingState.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# http://www.w3.org/TR/webrtc/#rtcpeerstate-enum +# https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection.signalingState#Value +z.calling.rtc.SignalingState = + NEW: 'new' + STABLE: 'stable' + LOCAL_OFFER: 'have-local-offer' + REMOTE_OFFER: 'have-remote-offer' + LOCAL_PROVISIONAL_ANSWER: 'have-local-pranswer' + REMOTE_PROVISIONAL_ANSWER: 'have-remote-pranswer' + CLOSED: 'closed' diff --git a/app/script/calling/rtc/StatsType.coffee b/app/script/calling/rtc/StatsType.coffee new file mode 100644 index 00000000000..9dbb43e8a07 --- /dev/null +++ b/app/script/calling/rtc/StatsType.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.calling ?= {} +z.calling.rtc ?= {} + +# http://www.w3.org/TR/webrtc/#idl-def-RTCStats +z.calling.rtc.StatsType = + CANDIDATE_PAIR: 'candidatepair' + GOOGLE_CANDIDATE_PAIR: 'googCandidatePair' + INBOUND_RTP: 'inboundrtp' + OUTBOUND_RTP: 'outboundrtp' + SSRC: 'ssrc' diff --git a/app/script/client/Client.coffee b/app/script/client/Client.coffee new file mode 100644 index 00000000000..da21ff436a4 --- /dev/null +++ b/app/script/client/Client.coffee @@ -0,0 +1,75 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +class z.client.Client + constructor: (payload) -> + # Preserved data from the backend + @[member] = payload[member] for member of payload + + # Maintained meta data by us + @meta = + is_verified: ko.observable false + primary_key: undefined + + @session = {} + + return @ + + ### + Splits an ID into user ID & client ID. + @param id [String] ID + @return [Object] Object containing the user ID & client ID + ### + @dismantle_user_client_id: (id) -> + id_parts = id?.split('@') or [] + return { + user_id: id_parts[0] + client_id: id_parts[1] + } + + ### + @return [Boolean] True, if the client is the self user's permanent client. + ### + is_permanent: -> + return @type is z.client.ClientType.PERMANENT + + ### + @return [Boolean] True, if it is NOT the client of the self user. + ### + is_remote: -> + return not @is_permanent() and not @is_temporary() + + ### + @return [Boolean] True, if the client is the self user's temporary client. + ### + is_temporary: -> + return @type is z.client.ClientType.TEMPORARY + + ### + This method returns a JSON object which can be stored in our local database. + + @return [JSON] JSON object + ### + to_json: -> + json = ko.toJSON @ + real_json = JSON.parse json + delete real_json.session + return real_json diff --git a/app/script/client/ClientError.coffee b/app/script/client/ClientError.coffee new file mode 100644 index 00000000000..6ace75a396b --- /dev/null +++ b/app/script/client/ClientError.coffee @@ -0,0 +1,41 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +class z.client.ClientError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @type = type + @stack = (new Error()).stack + + @:: = new Error() + @::constructor = @ + @::TYPE = { + CLIENT_NOT_SET: 'z.client.ClientError::TYPE.CLIENT_NOT_SET' + DATABASE_FAILURE: 'z.client.ClientError::TYPE.DATABASE_FAILURE' + MISSING_ON_BACKEND: 'z.client.ClientError::TYPE.MISSING_ON_BACKEND' + NO_CLIENT_ID: 'z.client.ClientError::TYPE.NO_CLIENT_ID' + NO_LOCAL_CLIENT: 'z.client.ClientError::TYPE.NO_LOCAL_CLIENT' + NO_USER_ID: 'z.client.ClientError::TYPE.NO_USER_ID' + REQUEST_FAILURE: 'z.client.ClientError::TYPE.REQUEST_FAILURE' + REQUEST_FORBIDDEN: 'z.client.ClientError::TYPE.REQUEST_FORBIDDEN' + TOO_MANY_CLIENTS: 'z.client.ClientError::TYPE.TOO_MANY_CLIENTS' + } diff --git a/app/script/client/ClientMapper.coffee b/app/script/client/ClientMapper.coffee new file mode 100644 index 00000000000..a8a48788e60 --- /dev/null +++ b/app/script/client/ClientMapper.coffee @@ -0,0 +1,60 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +class z.client.ClientMapper + + ### + Maps a JSON into a Client entity. + @param client_payload [JSON] Client payload + @return [z.client.Client] Client entity + ### + map_client: (client_payload) -> + client_et = new z.client.Client client_payload + + if client_payload.meta + client_et.meta.is_verified client_payload.meta.is_verified + client_et.meta.primary_key = client_payload.meta.primary_key + client_et.meta.user_id = (z.client.Client.dismantle_user_client_id client_payload.meta.primary_key).user_id + + return client_et + + ### + Maps an object of client IDs with their payloads to client entities. + @param clients_payload [Array] Client payloads + @return [Array] Array of client entities + ### + map_clients: (clients_payload) -> + return (@map_client client_payload for client_payload in clients_payload) + + ### + Update a client entity or object from JSON. + + @param client [z.client.Client or Object] Client + @param update_payload [JSON] JSON possibly containing updates + @return [Array] An array that contains the client and whether there was a change + ### + update_client: (client, update_payload) -> + contains_update = false + for member of update_payload + if client[member] isnt update_payload[member] + contains_update = true + client[member] = update_payload[member] + return [client, contains_update] diff --git a/app/script/client/ClientRepository.coffee b/app/script/client/ClientRepository.coffee new file mode 100644 index 00000000000..0f9f7fe88af --- /dev/null +++ b/app/script/client/ClientRepository.coffee @@ -0,0 +1,581 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +class z.client.ClientRepository + PRIMARY_KEY_CURRENT_CLIENT: 'local_identity' + constructor: (@client_service, @cryptography_repository) -> + @self_user = ko.observable undefined + @logger = new z.util.Logger 'z.client.ClientRepository', z.config.LOGGER.OPTIONS + + @client_mapper = new z.client.ClientMapper() + @clients = ko.observableArray() + @current_client = ko.observable undefined + + amplify.subscribe z.event.Backend.USER.CLIENT.ADD, @on_client_add + amplify.subscribe z.event.Backend.USER.CLIENT.REMOVE, @on_client_remove + amplify.subscribe z.event.WebApp.CLIENT.DELETE, @delete_client_and_session + + return @ + + init: (self_user) -> + @self_user self_user + @logger.log @logger.levels.INFO, "Initialized repository with user ID '#{@self_user().id}'" + + ############################################################################### + # Service interactions + ############################################################################### + + ### + Delete the temporary client on the backend. + @return [Promise] Promise that resolves when the temporary client was deleted on the backend + ### + delete_temporary_client: -> + return @client_service.delete_temporary_client @current_client().id + + ### + Load all known clients from the database. + @return [Promise] Promise that resolves with all the clients found in the local database + ### + get_all_clients_from_db: => + return @client_service.load_all_clients_from_db() + .then (clients) => + user_client_map = {} + for client in clients + ids = z.client.Client.dismantle_user_client_id client.meta.primary_key + continue if not ids.user_id or ids.user_id in [@self_user().id, @PRIMARY_KEY_CURRENT_CLIENT] + user_client_map[ids.user_id] ?= [] + client_et = @client_mapper.map_client client + client_et.session = @cryptography_repository.load_session ids.user_id, ids.client_id + user_client_map[ids.user_id].push client_et + return user_client_map + + ### + Retrieves meta information about specific client of the self user. + @param client_id [String] ID of client to be retrieved + @return [Promise] Promise that resolves with the retrieved client information + ### + get_client_by_id_from_backend: (client_id) => + @client_service.get_client_by_id client_id + + ### + Load all clients of a given user from the database. + @param user_id [String] ID of user to retrieve clients for + @return [Promise] Promise that resolves with all the known client entities for that user + ### + get_clients_from_db: (user_id) => + @client_service.load_clients_from_db_by_user_id user_id + .then (clients_payload) => + client_ets = @client_mapper.map_clients clients_payload + return client_ets + + ### + Loads a client from the database (if it exists). + @return [Promise] Promise that resolves with the local client + ### + get_current_client_from_db: => + @client_service.load_client_from_db @PRIMARY_KEY_CURRENT_CLIENT + .catch (error) -> + throw new z.client.ClientError error.message, z.client.ClientError::TYPE.DATABASE_FAILURE + .then (client_payload) => + if _.isString client_payload + error_message = "No current local client connected to '#{@PRIMARY_KEY_CURRENT_CLIENT}' found in database" + @logger.log @logger.levels.INFO, error_message + throw new z.client.ClientError error_message, z.client.ClientError::TYPE.NO_LOCAL_CLIENT + else + client_et = @client_mapper.map_client client_payload + @current_client client_et + @logger.log @logger.levels.INFO, + "Loaded local client '#{client_et.id}' connected to '#{@PRIMARY_KEY_CURRENT_CLIENT}'", @current_client() + return @current_client() + + ### + Updates properties for a client record in database. + + @todo Merge "meta" property before updating it, Object.assign(payload.meta, changes.meta) + @param user_id [String] User ID of the client owner + @param client_id [String] Client ID which needs to get updated + @param changes [String] New values which should be updated on the client + @return [Integer] Number of updated records + ### + update_client_in_db: (user_id, client_id, changes) -> + primary_key = @_construct_primary_key user_id, client_id + # Preserve primary key on update + changes.meta.primary_key = primary_key + return @client_service.update_client_in_db primary_key, changes + + ### + Construct the primary key to store clients in database. + @private + + @param user_id [String] User ID from the owner of the client + @param client_id [String] Client ID + @return [String] Primary key + ### + _construct_primary_key: (user_id, client_id) -> + throw new z.client.ClientError 'User ID is not defined', z.client.ClientError::TYPE.NO_USER_ID if not user_id + throw new z.client.ClientError 'Client ID is not defined', z.client.ClientError::TYPE.NO_CLIENT_ID if not client_id + return "#{user_id}@#{client_id}" + + ### + Save the a client into the database. + + @private + @param user_id [String] ID of user client to be stored belongs to + @param client_payload [Object] Client data to be stored in database + @return [Promise] Promise that resolves with the record stored in database + ### + _save_client: (user_id, client_payload) => + primary_key = @_construct_primary_key user_id, client_payload.id + return @client_service.save_client_in_db primary_key, client_payload + + ### + Save the local client into the database. + + @private + @param client_payload [Object] Client data to be stored in database + @return [Promise] Promise that resolves with the record stored in database + ### + _save_current_client: (client_payload) => + client_payload.meta = + is_verified: true + return @client_service.save_client_in_db @PRIMARY_KEY_CURRENT_CLIENT, client_payload + + + ############################################################################### + # Login and registration + ############################################################################### + + ### + Constructs the value for a cookie label. + @param login [String] Email or phone number of the user + @param client_type [z.client.ClientType] Temporary or permanent client type + ### + construct_cookie_label: (login, client_type) -> + login_hash = z.util.murmurhash3 login, 42 + client_type = @_load_current_client_type() if not client_type + return "webapp@#{login_hash}@#{client_type}@#{Date.now()}" + + ### + Constructs the key for a cookie label. + @param login [String] Email or phone number of the user + @param client_type [z.client.ClientType] Temporary or permanent client type + ### + construct_cookie_label_key: (login, client_type) -> + login_hash = z.util.murmurhash3 login, 42 + client_type = @_load_current_client_type() if not client_type + return "#{z.storage.StorageKey.AUTH.COOKIE_LABEL}@#{login_hash}@#{client_type}" + + ### + Validate existence of a local client online. + @param client [z.client.Client] Client retrieved from IndexedDB + @return [Promise] Promise that will resolve with an observable containing the client if valid + ### + get_valid_local_client: => + @get_current_client_from_db() + .then (client_et) => + return @get_client_by_id_from_backend client_et.id + .then (client) => + @logger.log @logger.levels.INFO, "Client with ID '#{client.id}' (#{client.type}) validated on backend" + return @current_client + .catch (error) => + client_et = @current_client() + @current_client undefined + error_message = error.code or error.message + + if error.code is z.service.BackendClientError::STATUS_CODE.NOT_FOUND + error_message = "Local client '#{client_et.id}' (#{client_et.type}) no longer exists on the backend" + @logger.log @logger.levels.WARN, error_message, error + @cryptography_repository.storage_repository.delete_everything() + .catch (error) => + error_message = "Deleting database after failed client validation unsuccessful: #{error.message}" + @logger.log @logger.levels.ERROR, error_message, error + throw new z.client.ClientError error_message, z.client.ClientError::TYPE.DATABASE_FAILURE + .then -> + throw new z.client.ClientError error_message, z.client.ClientError::TYPE.MISSING_ON_BACKEND + else if error.type is z.client.ClientError::TYPE.NO_LOCAL_CLIENT + throw error + else + @logger.log @logger.levels.ERROR, "Getting valid local client failed: #{error_message}", error + throw error + + ### + Register a new client. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/registerClient + + @note Password is needed for the registration of a client once 1st client has been registered. + + @param password [String] User password for verification + @return [Promise] Promise that will resolve with the newly registered client + ### + register_client: (password) => + return new Promise (resolve, reject) => + client_type = @_load_current_client_type() + + @cryptography_repository.generate_client_keys() + .then (keys) => + return @client_service.post_clients @_create_registration_payload client_type, password, keys + .catch (error) => + if error.label is z.service.BackendClientError::LABEL.TOO_MANY_CLIENTS + throw new z.client.ClientError error.message, z.client.ClientError::TYPE.TOO_MANY_CLIENTS + else + error_message = "Client registration request failed: #{error.message}" + @logger.log @logger.levels.ERROR, error_message, error + throw new z.client.ClientError error_message, z.client.ClientError::TYPE.REQUEST_FAILURE + .then (response) => + @logger.log @logger.levels.INFO, + "Registered '#{response.type}' client '#{response.id}' with cookie label '#{response.cookie}'", response + @current_client @client_mapper.map_client response + # Save client + return @_save_current_client response + .catch (error) => + if error.type in [z.client.ClientError::TYPE.REQUEST_FAILURE, z.client.ClientError::TYPE.TOO_MANY_CLIENTS] + throw error + else + error_message = "Failed to save client: #{error.message}" + @logger.log @logger.levels.ERROR, error_message, error + throw new z.client.ClientError error_message, z.client.ClientError::TYPE.DATABASE_FAILURE + .then (client_payload) => + # Update cookie + return @_transfer_cookie_label client_type, client_payload.cookie + .then => + resolve @current_client + .catch (error) => + @logger.log @logger.levels.ERROR, "Client registration failed: #{error.message}", error + reject error + + ### + Create payload for client registration. + + @private + @param client_type [z.client.ClientType] Type of client to be registered + @param password [String] User password + @param keys [Array] Array containing last resort key, pre-keys and signaling keys + @return [Object] Payload to register client with backend + ### + _create_registration_payload: (client_type, password, keys) -> + [last_resort_key, pre_keys, signaling_keys] = keys + + device_label = "#{platform.os.family}" + device_label += " #{platform.os.version}" if platform.os.version + device_model = platform.name + + if z.util.Environment.electron + if z.util.Environment.os.mac then identifier = z.string.wire_osx else identifier = z.string.wire_windows + device_model = z.localization.Localizer.get_text identifier + device_model = "#{device_model} (Internal)" if not z.util.Environment.frontend.is_production() + else + device_model = "#{device_model} (Temporary)" if client_type is z.client.ClientType.TEMPORARY + + return {} = + class: 'desktop' + cookie: @_get_cookie_label_value @self_user().email() or @self_user().phone() + label: device_label + lastkey: last_resort_key + model: device_model + password: password + prekeys: pre_keys + sigkeys: signaling_keys + type: client_type + + ### + Gets the value for a cookie label. + @private + @param login [String] Email or phone number of the user + ### + _get_cookie_label_value: (login) -> + return z.storage.get_value @construct_cookie_label_key login + + ### + Loads the cookie label value from the Local Storage and saves it into IndexedDB. + + @private + @param client_type [z.client.ClientType] Temporary or permanent client type + @param cookie_label [String] Cookie label, something like "webapp@2153234453@temporary@145770538393" + @return [Promise] Promise that resolves with the key of the stored cookie label + ### + _transfer_cookie_label: (client_type, cookie_label) => + indexed_db_key = z.storage.StorageKey.AUTH.COOKIE_LABEL + local_storage_key = @construct_cookie_label_key @self_user().email() or @self_user().phone(), client_type + + if cookie_label is undefined + cookie_label = @construct_cookie_label @self_user().email() or @self_user().phone(), client_type + @logger.log @logger.levels.WARN, "Cookie label is in an invalid state. We created a new one: '#{cookie_label}'" + z.storage.set_value local_storage_key, cookie_label + + @logger.log "Saving cookie label '#{cookie_label}' in IndexedDB", { + key: local_storage_key + value: cookie_label + } + + return @cryptography_repository.storage_repository.save_value indexed_db_key, cookie_label + + ### + Load current client type from amplify store. + @private + @return [z.client.ClientType] Type of current client + ### + _load_current_client_type: -> + return @current_client().type if @current_client() + is_permanent = z.storage.get_value z.storage.StorageKey.AUTH.PERSIST + type = if is_permanent then z.client.ClientType.PERMANENT else z.client.ClientType.TEMPORARY + type = if z.util.Environment.electron then z.client.ClientType.PERMANENT else type + return type + + + ############################################################################### + # Client handling + ############################################################################### + + ### + Delete client of a user on backend and removes it locally. + + @param client_id [String] ID of the client that should be deleted + @param password [String] Password entered by user + @return [Promise] Promise that resolves with the remaining user devices + ### + delete_client: (client_id, password) => + return new Promise (resolve, reject) => + if not password + error_message = "Could not delete client '#{client_id}' because password was not submitted" + @logger.log @logger.levels.ERROR, error_message + reject new z.client.ClientError error_message, z.client.ClientError::TYPE.REQUEST_FORBIDDEN + + @client_service.delete_client client_id, password + .then => + @_remove_client client_id + resolve @clients() + .catch (error) => + error_message = "Unable to delete client '#{client_id}': #{error.message}" + @logger.log @logger.levels.ERROR, error_message, + {error: error, password: password} + + if error.code is z.service.BackendClientError::STATUS_CODE.FORBIDDEN + error = new z.client.ClientError error_message, z.client.ClientError::TYPE.REQUEST_FORBIDDEN + else + error = new z.client.ClientError error_message, z.client.ClientError::TYPE.REQUEST_FAILURE + reject error + + ### + Delete a stored client and the session connected with it. + + @param user_id [String] ID of user + @param client_id [String] ID of client to be deleted + @return [Promise] Promise that resolves when a client and its session have been deleted + ### + delete_client_and_session: (user_id, client_id) => + @cryptography_repository.reset_session user_id, client_id + .then => + @delete_client_from_db user_id, client_id + + delete_client_from_db: (user_id, client_id) -> + primary_key = @_construct_primary_key user_id, client_id + return @client_service.delete_client_from_db primary_key + + ### + Retrieves meta information about all the clients of a given user. + @note If you want to get very detailed information about the devices from the own user, then use "@get_clients" + + @param user_id [String] User ID to retrieve client information for + @return [Promise] Promise that resolves with an array of client entities + ### + get_clients_by_user_id: (user_id) => + @client_service.get_clients_by_user_id user_id + .then (clients) => + return @_get_clients_by_user_id clients, user_id + + ### + Retrieves meta information about all the clients of the self user. + @param expect_current_client [Boolean] Should we check against the current local client + @return [Promise] Promise that resolves with the retrieved information about the clients + ### + get_clients_for_self: (expect_current_client = true) -> + @logger.log @logger.levels.INFO, "Retrieving all clients for the self user '#{@self_user().id}'" + @client_service.get_clients() + .then (response) => + return @_get_clients_by_user_id response, @self_user().id, expect_current_client + .then (client_ets) => + for possibly_new_client in client_ets + found = false + + @clients().forEach (client_et) -> + found = true if possibly_new_client.id is client_et.id + + @clients.push possibly_new_client if not found + @clients.sort (client_a, client_b) -> + return new Date(client_b.time) - new Date(client_a.time) + return @clients() + .catch (error) => + @logger.log @logger.levels.ERROR, "Unable to retrieve clients data: #{error}" + throw error + + ### + Is the current client permanent. + @return [Boolean] Type of current client is permanent + ### + is_current_client_permanent: => + throw new z.client.ClientError 'No current client', z.client.ClientError::TYPE.CLIENT_NOT_SET if not @current_client() + if z.util.Environment.electron + is_permanent = true + else + is_permanent = @current_client().is_permanent() + return is_permanent + + ### + Removes a client locally. + @param client_id [String] ID of the client that should be removed + @return [Promise] Promise that resolves with the primary key of the removed client + ### + remove_client: (client_id) -> + return new Promise (resolve, reject) => + user_id = @self_user().id + primary_key = @_construct_primary_key user_id, client_id + primary_key = @PRIMARY_KEY_CURRENT_CLIENT if @_is_current_client user_id, client_id + @client_service.delete_client_from_db primary_key + .then (primary_key) => + @clients.remove (client_et) -> + client_et.id is client_id + resolve primary_key + .catch (error) -> reject error + + ### + Match backend client response with locally stored ones. + @note: This function matches clients retrieved from the backend with the data stored in the local database. + Clients will then be updated with the backend payload in the database and mapped into entities. + + @private + @param clients [JSON] Payload from the backend + @param user_id [String] User ID + @param expect_current_client [Boolean] Should we check against the current local client + @return [Promise] Client entities + ### + _get_clients_by_user_id: (clients, user_id, expect_current_client) -> + return new Promise (resolve, reject) => + clients_from_backend = {} + clients_stored_in_db = [] + + client_keys = [] + + for client in clients + client_keys.push @_construct_primary_key user_id, client.id + clients_from_backend[client.id] = client + + # Find clients in database + @client_service.load_clients_from_db client_keys + .then (results) => + # Save new clients and cache existing ones + promises = [] + + # Updates a client payload if it does not fit the current database structure + update_client_schema = (user_id, client_payload) => + client_payload.meta = + is_verified: false + primary_key: @_construct_primary_key user_id, client_payload.id + return @_save_client user_id, client_payload + + # Known clients will be returned as object, unknown clients will resolve with their expected primary key + for result in results + # Handle new data which was not stored already in our local database + if _.isString result + ids = z.client.Client.dismantle_user_client_id result + if expect_current_client and @_is_current_client user_id, ids.client_id + @logger.log @logger.levels.INFO, "Current client '#{ids.client_id}' will not be changed in database" + continue + @logger.log @logger.levels.INFO, "Client '#{ids.client_id}' was not previously stored in database" + client_payload = clients_from_backend[ids.client_id] + promises.push update_client_schema user_id, client_payload + else + # Update existing clients with backend information + @logger.log @logger.levels.INFO, "Client '#{result.id}' was previously stored in database", result + [client_payload, contains_update] = @client_mapper.update_client result, clients_from_backend[result.id] + if contains_update + @logger.log @logger.levels.INFO, "Client '#{result.id}' will be overwritten with update in database", client_payload + promises.push @_save_client user_id, client_payload + else + clients_stored_in_db.push client_payload + + return Promise.all promises + .then (new_records) -> + # Cache new clients + return clients_stored_in_db.concat new_records + .then (all_clients) => + # Map clients to entities + client_ets = @client_mapper.map_clients all_clients + resolve client_ets + .catch (error) => + @logger.log @logger.levels.ERROR, "Unable to retrieve clients for user '#{user_id}': #{error.message}", error + reject error + + ### + Check if client is current local client. + + @private + @param user_id [String] User ID to be checked + @param client_id [String] ID of client to be checked + @return [Boolean] Is the client the current local client + ### + _is_current_client: (user_id, client_id) -> + throw new z.client.ClientError 'No current client', z.client.ClientError::TYPE.CLIENT_NOT_SET if not @current_client() + throw new z.client.ClientError 'User ID is not defined', z.client.ClientError::TYPE.NO_USER_ID if not user_id + throw new z.client.ClientError 'Client ID is not defined', z.client.ClientError::TYPE.NO_CLIENT_ID if not client_id + return user_id is @self_user().id and client_id is @current_client().id + + ### + Remove a client from the local clients. + @private + @param client_id [String] ID of client to be removed + ### + _remove_client: (client_id) => + for client in @clients() when client.id is client_id + @clients.remove client + break + + + ############################################################################### + # Conversation Events + ############################################################################### + + ### + A client was added by the self user. + @todo map, save and add to user + @param event_json [Object] JSON data of 'user.client-add' event + ### + on_client_add: (event_json) => + @logger.log @logger.levels.INFO, 'Client of self user added', event_json + amplify.publish z.event.WebApp.SELF.CLIENT_ADD, event_json.client + + ### + A client was added by the self user. + @param event_json [Object] JSON data of 'user.client-remove' event + ### + on_client_remove: (event_json) => + client_id = event_json?.client.id + @_client_removal client_id if client_id + + ### + Remove a client of the self user identified by id. + @private + @param client_id [String] ID of client to be removed + ### + _client_removal: (client_id) -> + @remove_client client_id + .then => + @logger.log "Removed client from database: #{client_id}" + amplify.publish z.event.WebApp.SIGN_OUT, 'client_removed', true if client_id is @current_client().id diff --git a/app/script/client/ClientService.coffee b/app/script/client/ClientService.coffee new file mode 100644 index 00000000000..0e4fd118dc9 --- /dev/null +++ b/app/script/client/ClientService.coffee @@ -0,0 +1,193 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +class z.client.ClientService + URL_CLIENTS: '/clients' + URL_USERS: '/users' + + constructor: (@client, @storage_service) -> + @logger = new z.util.Logger 'z.client.ClientService', z.config.LOGGER.OPTIONS + return @ + + + ############################################################################### + # Backend requests + ############################################################################### + + ### + Deletes a specific client from a user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/deleteClient + + @param client_id [String] ID of the client that should be deleted + @param password [String] Password entered by user + @return [Promise] Promise that resolves once the deletion of the client is complete + ### + delete_client: (client_id, password) -> + @client.send_json + url: @client.create_url "#{@URL_CLIENTS}/#{client_id}" + type: 'DELETE' + data: + password: password + + ### + Deletes the temporary client of a user. + @param client_id [String] ID of the temporary client to be deleted + @return [Promise] Promise that resolves once the deletion of the temporary client is complete + ### + delete_temporary_client: (client_id) -> + @client.send_json + url: @client.create_url "#{@URL_CLIENTS}/#{client_id}" + type: 'DELETE' + data: {} + + ### + Retrieves meta information about a specific client. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/getClients + + @param client_id [String] ID of client to be retrieved + @return [Promise] Promise that resolves with the requested client + ### + get_client_by_id: (client_id) -> + @client.send_request + url: @client.create_url "#{@URL_CLIENTS}/#{client_id}" + type: 'GET' + + ### + Retrieves meta information about all the clients self user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/listClients + @return [Promise] Promise that resolves with the clients of the self user + ### + get_clients: -> + @client.send_request + url: @client.create_url @URL_CLIENTS + type: 'GET' + + ### + Retrieves meta information about all the clients of a specific user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/getClients + + @param user_id [String] ID of user to retrieve clients for + @return [Promise] Promise that resolves with the clients of a user + ### + get_clients_by_user_id: (user_id) -> + @client.send_request + url: @client.create_url "#{@URL_USERS}/#{user_id}#{@URL_CLIENTS}" + type: 'GET' + + ### + Register a new client. + @param payload [Object] Client payload + @return [Promise] Promise that resolves with the registered client information + ### + post_clients: (payload) -> + @client.send_json + type: 'POST' + url: @client.create_url @URL_CLIENTS + data: payload + + + ############################################################################### + # Database requests + ############################################################################### + + ### + Removes a client from the database. + @param primary_key [String] Primary key used to find the client for deletion in the database + @return [Promise] Promise that resolves once the client is deleted + ### + delete_client_from_db: (primary_key) -> + return @storage_service.delete @storage_service.OBJECT_STORE_CLIENTS, primary_key + + ### + Load all clients we have stored in the database. + @return [Promise] Promise that resolves with all the clients payloads + ### + load_all_clients_from_db: => + return @storage_service.get_all @storage_service.OBJECT_STORE_CLIENTS + + ### + Loads a persisted client from the database. + @param primary_key [String] Primary key used to find a client in the database + @return [Promise] Promise that resolves with the client's payload or the primary key if not found + ### + load_client_from_db: (primary_key) -> + return new Promise (resolve, reject) => + @storage_service.db[@storage_service.OBJECT_STORE_CLIENTS] + .where 'meta.primary_key' + .equals primary_key + .first() + .then (client_record) => + if client_record is undefined + @logger.log @logger.levels.INFO, "Client with primary key '#{primary_key}' not found in database" + resolve primary_key + else + @logger.log @logger.levels.INFO, "Loaded client record from database '#{primary_key}'", client_record + resolve client_record + .catch (error) -> + reject error + + load_clients_from_db_by_user_id: (user_id) -> + return new Promise (resolve) => + store = @storage_service.OBJECT_STORE_CLIENTS + @storage_service.get_keys store, user_id + .then (primary_keys) => + return @load_clients_from_db primary_keys + .then (client_ets) -> + resolve client_ets + + ### + Loads persisted clients from the database. + @param primary_keys [Array] Primary keys used to find clients in the database + @return [Promise] Promise that resolves with the clients' payloads or the primary keys if not found + ### + load_clients_from_db: (primary_keys) -> + promises = (@load_client_from_db primary_key for primary_key in primary_keys) + return Promise.all promises + + ### + Persists a client. + + @param primary_key [String] Primary key used to find a client in the database + @param client_payload [JSON] Client payload + @return [Promise] Promise that resolves with the client payload stored in database + ### + save_client_in_db: (primary_key, client_payload) -> + client_payload.meta ?= {} + client_payload.meta.primary_key = primary_key + + return new Promise (resolve, reject) => + @storage_service.save @storage_service.OBJECT_STORE_CLIENTS, primary_key, client_payload + .then (primary_key) => + @logger.log @logger.levels.INFO, + "Client '#{client_payload.id}' stored with primary key '#{primary_key}'", client_payload + return @load_client_from_db primary_key + .then (record) -> resolve record + .catch (error) -> reject error + + ### + Updates a persisted client in the database. + + @param primary_key [String] Primary key used to find a client in the database + @param changes [JSON] Incremental update changes of the client JSON + @return [Promise] Number of updated records (1 if an object was updated, otherwise 0) + ### + update_client_in_db: (primary_key, changes) -> + return @storage_service.update @storage_service.OBJECT_STORE_CLIENTS, primary_key, changes diff --git a/app/script/client/ClientType.coffee b/app/script/client/ClientType.coffee new file mode 100644 index 00000000000..a0501a67d79 --- /dev/null +++ b/app/script/client/ClientType.coffee @@ -0,0 +1,24 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.client ?= {} + +z.client.ClientType = + PERMANENT: 'permanent' + TEMPORARY: 'temporary' diff --git a/app/script/components/accentColorPicker.coffee b/app/script/components/accentColorPicker.coffee new file mode 100644 index 00000000000..4820f12b4f1 --- /dev/null +++ b/app/script/components/accentColorPicker.coffee @@ -0,0 +1,58 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.AccentColorPicker + ### + Construct a new accent color picker view model. + + @param params [Object] + @option params [z.entity.User] user User entity + @option params [Object] selected Selected accent color + ### + constructor: (params) -> + + @user = ko.unwrap params.user + + @accent_colors = ko.computed => + [1..7].map (id) => + css_class = "accent-color-#{id}" + if @user? and @user.accent_id() is id + css_class += ' selected' + color = + css: css_class + id: id + + @on_select = (color) -> + params?.selected color + + +# Knockout registration of the accent color picker component. +ko.components.register 'accent-color-picker', + viewModel: z.components.AccentColorPicker + template: """ + +
    +
    +
    +
    +
    + + """ diff --git a/app/script/components/asset/audioAsset.coffee b/app/script/components/asset/audioAsset.coffee new file mode 100644 index 00000000000..c69ffbec6d5 --- /dev/null +++ b/app/script/components/asset/audioAsset.coffee @@ -0,0 +1,116 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.AudioAssetComponent + ### + Construct a new audio asset. + + @param params [Object] + @option params [ko.observableArray] asset + ### + constructor: (params, component_info) -> + @logger = new z.util.Logger 'AudioAssetComponent', z.config.LOGGER.OPTIONS + @asset = params.asset + + @audio_src = ko.observable() + @audio_element = $(component_info.element).find('audio')[0] + @audio_time = ko.observable 0 + @audio_is_loaded = ko.observable false + + @show_loudness_preview = ko.pureComputed => + return @asset.meta?.loudness?.length > 0 + + if @asset.meta? + @audio_time @asset.meta.duration + + $(component_info.element).attr + 'data-uie-name': 'audio-asset' + 'data-uie-value': @asset.file_name + + on_loadedmetadata: => + @_send_tracking_event() + + on_timeupdate: => + @audio_time @audio_element.currentTime + + on_play_button_clicked: => + if @audio_src()? + @audio_element?.play() + else + @asset.load() + .then (blob) => + @audio_src window.URL.createObjectURL blob + @audio_is_loaded true + @audio_element?.play() + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to load audio asset ', error + + on_pause_button_clicked: => + @audio_element?.pause() + + _send_tracking_event: => + duration = Math.floor @audio_element.duration + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.PLAYED_AUDIO_MESSAGE, + duration: z.util.bucket_values(duration, [0, 10, 30, 60, 300, 900, 1800]) + duration_actual: duration + type: z.util.get_file_extension @asset.file_name + + +ko.components.register 'audio-asset', + viewModel: createViewModel: (params, component_info) -> + return new z.components.AudioAssetComponent params, component_info + template: """ + + +
    +
    + + + +
    +
    + + + + + + + + + + + + + + + + """ diff --git a/app/script/components/asset/controls/audioSeekBar.coffee b/app/script/components/asset/controls/audioSeekBar.coffee new file mode 100644 index 00000000000..ce88147cfa5 --- /dev/null +++ b/app/script/components/asset/controls/audioSeekBar.coffee @@ -0,0 +1,100 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.AudioSeekBarComponent + ### + Construct a audio seek bar that renders audio levels + + @param params [Object] + @option src [HTMLElement] media src + @option asset [z.entity.File] + @option disabled [Boolean] + ### + constructor: (params, component_info) -> + @audio_element = params.src + @asset = params.asset + + @element = component_info.element + @loudness = [] + + @disabled = ko.computed => + is_disabled = params.disabled?() + $(@element).toggleClass 'element-disabled', is_disabled + + if @asset.meta?.loudness?.length + @loudness = @_normalize_loudness @asset.meta.loudness, component_info.element.clientHeight + + @_on_resize_fired = _.debounce => + @_render_levels() + @_on_time_update() + , 500 + + @_render_levels() + + @audio_element.addEventListener 'ended', @_on_audio_ended + @audio_element.addEventListener 'timeupdate', @_on_time_update + component_info.element.addEventListener 'click', @_on_level_click + window.addEventListener 'resize', @_on_resize_fired + + _render_levels: => + number_of_levels_fit_on_screen = Math.floor @element.clientWidth / 3 # 2px + 1px + scaled_loudness = z.util.ArrayUtil.interpolate @loudness, number_of_levels_fit_on_screen + + $(@element).empty() + $('').height(level).appendTo(@element) for level in scaled_loudness + + _normalize_loudness: (loudness, max) -> + peak = Math.max.apply Math, loudness + return if peak > max then loudness.map (level) -> level * max / peak else loudness + + _on_level_click: (e) => + mouse_x = e.pageX - $(e.currentTarget).offset().left + @audio_element.currentTime = @audio_element.duration * mouse_x / e.currentTarget.clientWidth + @_on_time_update() + + _on_time_update: => + $levels = @_clear_theme() + index = Math.floor @audio_element.currentTime / @audio_element.duration * $levels.length + @_add_theme index + + _on_audio_ended: => + @_clear_theme() + + _clear_theme: => + $(@element).children().removeClass 'bg-theme' + + _add_theme: (index) => + $(@element).children() + .eq index + .prevAll().addClass 'bg-theme' + + dispose: => + @disabled.dispose() + @audio_element.removeEventListener 'ended', @_on_audio_ended + @audio_element.removeEventListener 'timeupdate', @_on_time_update + @element.removeEventListener 'click', @_on_level_click + window.removeEventListener 'resize', @_on_resize_fired + + +ko.components.register 'audio-seek-bar', + viewModel: createViewModel: (params, component_info) -> + return new z.components.AudioSeekBarComponent params, component_info + template: """""" diff --git a/app/script/components/asset/controls/mediaButton.coffee b/app/script/components/asset/controls/mediaButton.coffee new file mode 100644 index 00000000000..c2a83db8cad --- /dev/null +++ b/app/script/components/asset/controls/mediaButton.coffee @@ -0,0 +1,85 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.MediaButtonComponent + ### + Construct a media button. + + @param params [Object] + @option src [HTMLElement] media src + @option large [Boolean] display large button + @option asset [z.entity.File] + ### + constructor: (params, component_info) -> + @media_element = params.src + @large = params.large + @asset = params.asset + + if @large + component_info.element.classList.add 'media-button-lg' + + @media_is_playing = ko.observable false + + @svg_view_box = ko.pureComputed => + size = if @large then 64 else 32 + return "0 0 #{size} #{size}" + + @circle_upload_progress = ko.pureComputed => + size = if @large then '200' else '100' + return "#{@asset.upload_progress() * 2} #{size}" + + @circle_download_progress = ko.pureComputed => + size = if @large then '200' else '100' + return "#{@asset.download_progress() * 2} #{size}" + + @on_play_button_clicked = -> params.play?() + @on_pause_button_clicked = -> params.pause?() + @on_cancel_button_clicked = -> params.cancel?() + + @media_element.addEventListener 'playing', => @media_is_playing true + @media_element.addEventListener 'pause', => @media_is_playing false + + +ko.components.register 'media-button', + viewModel: createViewModel: (params, component_info) -> + return new z.components.MediaButtonComponent params, component_info + template: """ + +
    +
    + + +
    +
    + + + +
    + + +
    +
    + + + +
    + + """ diff --git a/app/script/components/asset/controls/seekBar.coffee b/app/script/components/asset/controls/seekBar.coffee new file mode 100644 index 00000000000..979fdc82d72 --- /dev/null +++ b/app/script/components/asset/controls/seekBar.coffee @@ -0,0 +1,86 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.SeekBarComponent + ### + Construct a seek bar. + + @param params [Object] + @option src [HTMLElement] media src + ### + constructor: (params, component_info) -> + @media_element = params.src + @dark_mode = params.dark + @disabled = ko.pureComputed -> params.disabled?() + + @seek_bar = $(component_info.element).find('input')[0] + @seek_bar_mouse_over = ko.observable false + @seek_bar_thumb_dragged = ko.observable false + + @show_seek_bar_thumb = ko.pureComputed => + return @seek_bar_thumb_dragged() or @seek_bar_mouse_over() + + @seek_bar.addEventListener 'mousedown', => + @media_element.pause() + @seek_bar_thumb_dragged true + + @seek_bar.addEventListener 'mouseup', => + @media_element.play() + @seek_bar_thumb_dragged false + + @seek_bar.addEventListener 'mouseenter', => + @seek_bar_mouse_over true + + @seek_bar.addEventListener 'mouseleave', => + @seek_bar_mouse_over false + + @seek_bar.addEventListener 'change', => + time = @media_element.duration * (@seek_bar.value / 100) + @media_element.currentTime = time + + @media_element.addEventListener 'timeupdate', => + value = (100 / @media_element.duration) * @media_element.currentTime + @_update_seek_bar value + + @media_element.addEventListener 'ended', => + @_update_seek_bar 100 + + @_update_seek_bar_style 0 + + _update_seek_bar: (progress) => + return if @media_element.paused and progress < 100 + @seek_bar.value = progress + @_update_seek_bar_style progress + + _update_seek_bar_style: (progress) => + # TODO check if we can find a css solution + if @dark_mode + @seek_bar.style.backgroundImage = "linear-gradient(to right, currentColor #{progress}%, rgba(141,152,159,0.24) #{progress}%)" + else + @seek_bar.style.backgroundImage = "linear-gradient(to right, currentColor #{progress}%, rgba(255,255,255,0.4) #{progress}%)" + + +ko.components.register 'seek-bar', + viewModel: createViewModel: (params, component_info) -> + return new z.components.SeekBarComponent params, component_info + template: """ + + """ diff --git a/app/script/components/asset/fileAsset.coffee b/app/script/components/asset/fileAsset.coffee new file mode 100644 index 00000000000..ef4590bb92c --- /dev/null +++ b/app/script/components/asset/fileAsset.coffee @@ -0,0 +1,112 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.FileAssetComponent + ### + Construct a new audio asset. + + @param params [Object] + @option params [ko.observableArray] asset + ### + constructor: (params, component_info) -> + @asset = params.asset + + @circle_upload_progress = ko.pureComputed => + size = if @large then '200' else '100' + return "#{@asset.upload_progress() * 2} #{size}" + + @circle_download_progress = ko.pureComputed => + size = if @large then '200' else '100' + return "#{@asset.download_progress() * 2} #{size}" + + @file_extension = ko.pureComputed => + ext = z.util.get_file_extension @asset.file_name + return if ext.length <= 3 then ext else '' + + +ko.components.register 'file-asset', + viewModel: createViewModel: (params, component_info) -> + return new z.components.FileAssetComponent params, component_info + template: """ +
    + +
    +
    + + + +
    +
    + + + +
    + +
    + + +
    +
    +
    + + + +
    + + +
    +
    +
    + + + +
    + + +
    + +
    +
    +
      +
    • + +
    • + + +
    • + + +
    • + + +
    • + +
    +
    + +
    + """ diff --git a/app/script/components/asset/linkPreviewAsset.coffee b/app/script/components/asset/linkPreviewAsset.coffee new file mode 100644 index 00000000000..9760a6c66d9 --- /dev/null +++ b/app/script/components/asset/linkPreviewAsset.coffee @@ -0,0 +1,60 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.LinkPreviewAssetComponent + ### + Construct a new audio asset. + + @param params [Object] + @option params [z.entity.LinkPreview] preview + ### + constructor: (params, component_info) -> + @preview = params.preview + @viewport_changed = params.viewport_changed + @element = component_info.element + + on_link_preview_click: => + window.open @preview.permanent_url + + dispose: => + @element.removeEventListener 'click', @on_link_preview_click + + +ko.components.register 'link-preview-asset', + viewModel: createViewModel: (params, component_info) -> + return new z.components.LinkPreviewAssetComponent params, component_info + template: """ + + + + + + +
    + + + +
    +
    + + """ diff --git a/app/script/components/asset/locationAsset.coffee b/app/script/components/asset/locationAsset.coffee new file mode 100644 index 00000000000..680ef003d49 --- /dev/null +++ b/app/script/components/asset/locationAsset.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.LocationAssetComponent + ### + Construct a new audio asset. + + @param params [Object] + @option params [ko.observableArray] asset + ### + constructor: (params) -> + @asset = params.asset + + +ko.components.register 'location-asset', + viewModel: z.components.LocationAssetComponent + template: """ +
    +
    + + """ diff --git a/app/script/components/asset/videoAsset.coffee b/app/script/components/asset/videoAsset.coffee new file mode 100644 index 00000000000..89ad6cb1b5d --- /dev/null +++ b/app/script/components/asset/videoAsset.coffee @@ -0,0 +1,131 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.VideoAssetComponent + ### + Construct a new video asset. + + @param params [Object] + @option asset [z.entity.File] + ### + constructor: (params, component_info) -> + @logger = new z.util.Logger 'VideoAssetComponent', z.config.LOGGER.OPTIONS + @asset = params.asset + + @video_element = $(component_info.element).find('video')[0] + @video_src = ko.observable() + @video_time = ko.observable() + + @video_playback_error = ko.observable false + @show_bottom_controls = ko.observable false + + @video_time_rest = ko.pureComputed => + return @video_element.duration - @video_time() + + if @asset.preview_resource() + @_load_video_preview() + else + @asset.preview_resource.subscribe @_load_video_preview + + _load_video_preview: => + @asset.load_preview() + .then (blob) => + @video_element.setAttribute 'poster', window.URL.createObjectURL blob + @video_element.style.backgroundColor = '#000' + + on_loadedmetadata: => + @video_time @video_element.duration + @_send_tracking_event() + + on_timeupdate: => + @video_time @video_element.currentTime + + on_error: => + @video_playback_error true + + on_play_button_clicked: => + if @video_src()? + @video_element?.play() + else + @asset.load() + .then (blob) => + @video_src window.URL.createObjectURL blob + @video_element?.play() + @show_bottom_controls true + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to load video asset ', error + + on_pause_button_clicked: => + @video_element?.pause() + + on_video_playing: => + @video_element.style.backgroundColor = '#000' + + _send_tracking_event: => + duration = Math.floor @video_element.duration + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.PLAYED_VIDEO_MESSAGE, + duration: z.util.bucket_values(duration, [0, 10, 30, 60, 300, 900, 1800]) + duration_actual: duration + +ko.components.register 'video-asset', + viewModel: createViewModel: (params, component_info) -> + return new z.components.VideoAssetComponent params, component_info + template: """ +
    + + +
    + + + +
    +
    + + + +
    +
    + + +
    + + +
    +
    + + +
    + + +
    + """ diff --git a/app/script/components/calling/chooseScreen.coffee b/app/script/components/calling/chooseScreen.coffee new file mode 100644 index 00000000000..84c4a732ac7 --- /dev/null +++ b/app/script/components/calling/chooseScreen.coffee @@ -0,0 +1,45 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.ChooseScreen + constructor: (params) -> + @on_cancel = params.cancel + @on_choose = params.choose + @screens = params.screens or [] + + +ko.components.register 'choose-screen', + viewModel: z.components.ChooseScreen + template: """ +
    + +
    + +
    + +
    +
    +
    +
    +
    + """ diff --git a/app/script/components/calling/deviceToggleButton.coffee b/app/script/components/calling/deviceToggleButton.coffee new file mode 100644 index 00000000000..55b5b31a64c --- /dev/null +++ b/app/script/components/calling/deviceToggleButton.coffee @@ -0,0 +1,38 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.DeviceToggleButton + constructor: (params) -> + @current_device_index = params.index + @number_of_devices = params.length + @icon_class = if params.type is z.calling.enum.MediaDeviceType.VIDEO_INPUT then 'icon-video' else 'icon-screensharing' + + +ko.components.register 'device-toggle-button', + viewModel: z.components.DeviceToggleButton + template: """ +
    +
    + + + +
    + """ diff --git a/app/script/components/commonContacts.coffee b/app/script/components/commonContacts.coffee new file mode 100644 index 00000000000..7df158bea3b --- /dev/null +++ b/app/script/components/commonContacts.coffee @@ -0,0 +1,64 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.CommonContactsViewModel + constructor: (params, component_info) -> + # parameter list + @user = ko.unwrap params.user + @element = component_info.element + @search_repository = wire.app.repository.search + + @displayed_contacts = ko.observableArray() + @more_contacts = ko.observable 0 + @_set_contacts_data 0 + + @search_repository.get_common_contacts @user.id + .then (user_ets) => + @_display_contacts user_ets + .catch (error) => + @logger.log @logger.levels.ERROR, "Could not update users common contacts: #{error.message}", error + + _set_contacts_data: (value) -> + $(@element).attr 'data-contacts', value + + _display_contacts: (contacts) => + number_of_contacts = contacts.length + number_to_show = if number_of_contacts is 4 then 4 else 3 + @displayed_contacts contacts.slice 0, number_to_show + @more_contacts number_of_contacts - number_to_show + @_set_contacts_data number_of_contacts + + +ko.components.register 'common-contacts', + viewModel: createViewModel: (params, component_info) -> + return new z.components.CommonContactsViewModel params, component_info + template: """ +
    +
    +
    + +
    +
    +
    + + + + """ diff --git a/app/script/components/deviceCard.coffee b/app/script/components/deviceCard.coffee new file mode 100644 index 00000000000..66f5a93f6e7 --- /dev/null +++ b/app/script/components/deviceCard.coffee @@ -0,0 +1,85 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.DeviceCard + constructor: (params, component_info) -> + @device = ko.unwrap params.device + @id = @device?.id or '' + @label = @device?.label or '?' + @model = @device?.model or @device?.class or '?' # devices for other users will only provide the device class + @class = @device?.class or '?' + + @current = params.current or false + @detailed = params.detailed or false + @click = params.click + + @data_uie_name = 'device-card-info' + @data_uie_name += '-current' if @current + + @location = ko.pureComputed => + result = ko.observable '?' + z.location.get_location @device.location?.lat, @device.location?.lon, (error, location) -> + result "#{location.place}, #{location.country_code}" if location + return result + + $(component_info.element).addClass 'device-card-no-hover' if @detailed or not @click + $(component_info.element).addClass 'device-card-detailed' if @detailed + + on_click_device: => + @click? @device + + print_time: (timestamp) -> + reg_moment = moment(timestamp) + reg_format = if moment().year() is reg_moment.year() then 'ddd D MMM, HH:mm' else 'ddd D MMM YYYY, HH:mm' + return reg_moment.format reg_format + + +ko.components.register 'device-card', + viewModel: + createViewModel: (params, component_info) -> + return new z.components.DeviceCard params, component_info + template: """ +
    + +
    + + +
    +
    + + +
    + + +
    +
    + + +
    +
    + + +
    +
    + +
    + """ diff --git a/app/script/components/deviceRemove.coffee b/app/script/components/deviceRemove.coffee new file mode 100644 index 00000000000..bb596b648ed --- /dev/null +++ b/app/script/components/deviceRemove.coffee @@ -0,0 +1,69 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.DeviceRemove + constructor: (params, component_info) -> + @device = ko.unwrap params.device + @device_remove_error = params.error or ko.observable false + @model = @device.model + + @remove_form_visible = ko.observable false + + @password = ko.observable '' + @password.subscribe (value) => + @device_remove_error false if value.length > 0 + + @click_on_submit = => + params.remove? @password(), @device + + @click_on_cancel = => + @remove_form_visible false + params.cancel?() + + @click_on_remove_device = => + @remove_form_visible true + + +ko.components.register 'device-remove', + viewModel: createViewModel: (params, component_info) -> + return new z.components.DeviceRemove params, component_info + template: """ + + + + +
    + + + +
    + + """ diff --git a/app/script/components/groupList.coffee b/app/script/components/groupList.coffee new file mode 100644 index 00000000000..2985e45fa19 --- /dev/null +++ b/app/script/components/groupList.coffee @@ -0,0 +1,52 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.GroupListViewModel + ### + Construct a new group list view model. + + @param params [Object] + @option params [ko.observableArray] groups Data source + @option params [Function] click Function called when a list item is clicked + ### + constructor: (params) -> + # parameter list + @groups = params.groups + @avatar = params.avatar or false + @on_select = params.click + + +# Knockout registration of the group list component. +ko.components.register 'group-list', + viewModel: z.components.GroupListViewModel + template: """ +
    +
    + + + + +
    + +
    +
    +
    + """ diff --git a/app/script/components/inputElement.coffee b/app/script/components/inputElement.coffee new file mode 100644 index 00000000000..b11d85255dd --- /dev/null +++ b/app/script/components/inputElement.coffee @@ -0,0 +1,56 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.InputElement + constructor: (params, component_info) -> + + @value = params.value + + @change = (data, event) => + new_name = z.util.remove_line_breaks event.target.value.trim() + old_name = @value().trim() + event.target.value = old_name + @editing false + params.change? new_name if new_name isnt old_name + + @edit = -> @editing true + + @editing = ko.observable false + + @editing.subscribe (value) => + if value + $(component_info.element).find('textarea').one 'keydown', (e) => + @editing false if e.keyCode is z.util.KEYCODE.ESC + else + $(component_info.element).find('textarea').off 'keydown', 'esc', @abort + + @placeholder = params.placeholder + + +# Knockout registration of the input element component. +ko.components.register 'input-element', + viewModel: + createViewModel: (params, component_info) -> + return new z.components.InputElement params, component_info + template: """ + + + """ diff --git a/app/script/components/topPeople.coffee b/app/script/components/topPeople.coffee new file mode 100644 index 00000000000..0a2b13e8f1c --- /dev/null +++ b/app/script/components/topPeople.coffee @@ -0,0 +1,49 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.TopPeopleViewModel + constructor: (params) -> + @user_ets = params.user + @user_selected = params.selected + @max_users = params.max or 9 + + @displayed_users = ko.pureComputed => + return @user_ets().slice 0, @max_users + + @on_select = (user_et) => + if @is_selected user_et then @user_selected.remove user_et else @user_selected.push user_et + + @is_selected = (user_et) => + return user_et in @user_selected() + + +ko.components.register 'top-people', + viewModel: z.components.TopPeopleViewModel + template: """ +
    +
    + +
    +
    +
    +
    +
    + """ diff --git a/app/script/components/userAvatar.coffee b/app/script/components/userAvatar.coffee new file mode 100644 index 00000000000..c1289bc0478 --- /dev/null +++ b/app/script/components/userAvatar.coffee @@ -0,0 +1,89 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.UserAvatar + constructor: (params, component_info) -> + @user = ko.unwrap params.user + @badge = params.badge or false + @element = $(component_info.element) + + if not @user? + @user = new z.entity.User() + + @element.attr + 'id': z.util.create_random_uuid() + 'user-id': @user.id + + @initials = ko.computed => + if @element.hasClass 'user-avatar-xs' + return z.util.get_first_character @user.initials() + else + return @user.initials() + + @state = ko.computed => + status = @user.connection().status() + return 'self' if @user.is_me + return 'selected' if params.selected?() is true + return 'blocked' if status is z.user.ConnectionStatus.BLOCKED + return 'pending' if status in [z.user.ConnectionStatus.SENT, z.user.ConnectionStatus.PENDING] + return 'ignored' if status is z.user.ConnectionStatus.IGNORED + return 'unknown' if status in [z.user.ConnectionStatus.UNKNOWN, z.user.ConnectionStatus.CANCELLED] + return '' + + @css_classes = ko.computed => + class_string = "accent-color-#{@user.accent_id()}" + class_string += " #{@state()}" if @state() + return class_string + + @on_click = (data, event) -> + params.click? data.user, event.currentTarget.parentNode + + ko.computed => + image_url = @user.picture_preview_url() + image_was_already_loaded = false + + if image_url? + image = new Image() + image.onload = => + @avatar_image = @element.find '.user-avatar-image' + @avatar_image.empty().append image + + requestAnimFrame => + if not image_was_already_loaded + @element.addClass 'user-avatar-loading-transition' + @element.addClass 'user-avatar-image-loaded' + + image.src = z.util.strip_url_wrapper image_url + image_was_already_loaded = image.complete + +ko.components.register 'user-avatar', + viewModel: + createViewModel: (params, component_info) -> + return new z.components.UserAvatar params, component_info + template: """ +
    +
    +
    +
    +
    +
    +
    + """ diff --git a/app/script/components/userInput.coffee b/app/script/components/userInput.coffee new file mode 100644 index 00000000000..4446da1545f --- /dev/null +++ b/app/script/components/userInput.coffee @@ -0,0 +1,67 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +class z.components.UserListInputViewModel + constructor: (params, component_info) -> + @input = params.input + @selected = params.selected or ko.observableArray [] + @placeholder = params.placeholder + @on_enter = params.enter + @on_close = params.close + + @element = component_info.element + @input_element = $(@element).find '.search-input' + @inner_element = $(@element).find '.search-inner' + + @selected.subscribe => + @input '' + @input_element.focus() + setTimeout => + @inner_element.scrollTop @inner_element[0].scrollHeight + + @placeholder = ko.computed => + if @input() is '' and @selected().length is 0 + return z.localization.Localizer.get_text params.placeholder + else + return '' + + on_key_press: (data, event) => + @selected.pop() if event.keyCode is z.util.KEYCODE.DELETE and @input() is '' + return true + + +ko.components.register 'user-input', + viewModel: + createViewModel: (params, component_info) -> + return new z.components.UserListInputViewModel params, component_info + template: """ +
    +
    +
    + + + + +
    +
    +
    +
    + """ diff --git a/app/script/components/userList.coffee b/app/script/components/userList.coffee new file mode 100644 index 00000000000..e8ea0e5f805 --- /dev/null +++ b/app/script/components/userList.coffee @@ -0,0 +1,142 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + +z.components.UserListMode = + DEFAULT: 'default' + COMPACT: 'compact' + INFO: 'info' + +# displays a list of user_ets +# +# @param [Object] params to adjust user list +# @option params [ko.observableArray] user data source +# @option params [ko.observable] filter filter list items +# @option params [function] click is called when a list item is selected +# @option params [function] connect is called when the connect button is clicked +# @option params [function] dismiss is called when the dismiss button is clicked +# @option params [ko.observable] selected will be populated will all the selected items +# @option params [function] selected_filter that determines if the user can be selected +# +class z.components.UserListViewModel + constructor: (params) -> + # parameter list + @user_ets = params.user + @user_click = params.click + @user_connect = params.connect + @user_dismiss = params.dismiss + @user_filter = params.filter + @user_selected = params.selected + @user_selected_filter = params.selectable + @mode = params.mode or z.components.UserListMode.DEFAULT + + @css_classes = ko.computed => + if @mode is z.components.UserListMode.COMPACT + return 'search-list-sm' + else if @mode is z.components.UserListMode.INFO + return 'search-list-lg' + else + return 'search-list-md' + + @show_buttons = => + return @user_connect? + + # defaults + @filtered_user_ets = @user_ets + @is_selected = -> return false + @is_selectable = -> return true + @on_select = (user_et, e) => @user_click? user_et, e + @on_dismiss = (user_et, e) => + e.stopPropagation() + @user_dismiss? user_et, e + @on_connect = (user_et, e) => + e.stopPropagation() + @user_connect? user_et, e + + # filter all list items if a filter is provided + if @user_filter? + @filtered_user_ets = ko.computed => + ko.utils.arrayFilter @user_ets(), (user_et) => + user_name = window.getSlug user_et.name() + search_query = window.getSlug @user_filter() + matches_name = z.util.contains user_name, search_query + matches_email = user_et.email() is @user_filter() + return matches_name or matches_email + + # check every list item before selection if selected_filter is provided + if @user_selected_filter? + @is_selectable = @user_selected_filter + + # list will be selectable if select is provided + if @user_selected? + @on_select = (user_et) => + is_selected = @is_selected user_et + if is_selected + @user_selected.remove user_et + else + @user_selected.push user_et if @is_selectable user_et + + @user_click? user_et, not is_selected + + @is_selected = (user_et) => + return user_et in @user_selected() + + +ko.components.register 'user-list', + viewModel: z.components.UserListViewModel + template: """ +
    +
    + + +
    +
    +
    + + +
    + +
    +
    +
    +
    +
    +
    +
    + + + +
    +
    + + +
    + +
    +
    + + +
    + + +
    + + + """ diff --git a/app/script/components/userProfile.coffee b/app/script/components/userProfile.coffee new file mode 100644 index 00000000000..39218d9ac82 --- /dev/null +++ b/app/script/components/userProfile.coffee @@ -0,0 +1,269 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.components ?= {} + + +z.components.UserProfileMode = + DEFAULT: 'default' + PEOPLE: 'people' + SEARCH: 'search' + +class z.components.UserProfileViewModel + constructor: (params, component_info) -> + @logger = new z.util.Logger 'z.components.UserProfileViewModel', z.config.LOGGER.OPTIONS + + @user = params.user + @conversation = params.conversation + @mode = params.mode or z.components.UserProfileMode.DEFAULT + + # repository references + @client_repository = wire.app.repository.client + @conversation_repository = wire.app.repository.conversation + @cryptography_repository = wire.app.repository.cryptography + @user_repository = wire.app.repository.user + + # component dom element + @element = $(component_info.element) + + # actions + @on_accept = -> params.accept? @user() + @on_add_people = => params.add_people? @user() + @on_block = -> params.block? @user() + @on_close = -> params.close?() + @on_ignore = -> params.ignore? @user() + @on_leave = => params.leave? @user() + @on_profile = => params.profile? @user() + @on_remove = => params.remove? @user() + @on_unblock = -> params.unblock? @user() + + # cancel request confirm dialog + @confirm_dialog = undefined + + # tabs + @click_on_tab = (index) => @tab_index index + @tab_index = ko.observable 0 + @tab_index.subscribe @on_tab_index_changed + + # devices + @devices = ko.observableArray() + @devices_found = ko.observable() + @selected_device = ko.observable() + @fingerprint_remote = ko.observable '' + @fingerprint_local = ko.observable '' + @is_resetting_session = ko.observable false + + # destroy confirm dialog when user changes + ko.computed => + @confirm_dialog?.destroy() if @user()? + @tab_index 0 + @devices_found null + @selected_device null + @fingerprint_remote '' + @is_resetting_session false + + @selected_device.subscribe => + if @selected_device()? + @cryptography_repository.get_session @user().id, @selected_device().id + .then (cryptobox_session) => + @fingerprint_remote cryptobox_session.fingerprint_remote() + @fingerprint_local cryptobox_session.fingerprint_local() + + @add_people_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_people_add + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.ADD_PEOPLE} + } + + @device_headline = ko.pureComputed => + z.localization.Localizer.get_text { + id: z.string.people_tabs_devices_headline + replace: {placeholder: '%@.name', content: @user().first_name()} + } + + @no_device_headline = ko.pureComputed => + z.localization.Localizer.get_text { + id: z.string.people_tabs_no_devices_headline + replace: {placeholder: '%@.name', content: @user().first_name()} + } + + @detail_message = ko.pureComputed => + z.localization.Localizer.get_text { + id: z.string.people_tabs_device_detail_headline + replace: [ + {placeholder: '%bold', content: "'} + ] + } + + @on_cancel_request = => + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + @confirm_dialog = @element.confirm + template: '#template-confirm-cancel_request' + data: + user: @user() + confirm: => + should_block = @element.find('.checkbox input').is ':checked' + if should_block + @user_repository.block_user @user() + else + @user_repository.cancel_connection_request @user() + + conversation_et = @conversation_repository.get_one_to_one_conversation @user().id + if @conversation_repository.is_active_conversation conversation_et + amplify.publish z.event.WebApp.CONVERSATION.PEOPLE.HIDE + next_conversation_et = @conversation_repository.get_next_conversation conversation_et + setTimeout -> + amplify.publish z.event.WebApp.CONVERSATION.SHOW, next_conversation_et + , 550 + + params.cancel_request? @user() + + @on_open = => + amplify.publish z.event.WebApp.CONVERSATION.PEOPLE.HIDE + conversation_et = @conversation_repository.get_one_to_one_conversation @user().id + @conversation_repository.unarchive_conversation conversation_et if conversation_et.is_archived() + setTimeout => + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + params.open? @user() + , 550 + + @on_connect = => + @user_repository.create_connection @user(), true + .then -> + amplify.publish z.event.WebApp.CONVERSATION.PEOPLE.HIDE + + params.connect? @user() + + @on_pending = => + if @user().connection().status() in [z.user.ConnectionStatus.PENDING, z.user.ConnectionStatus.IGNORED] + params.pending? @user() + else + @on_open() + + @accent_color = ko.computed => + return "accent-color-#{@user()?.accent_id()}" + , @, deferEvaluation: true + + @show_gray_image = ko.computed => + return false if not @user()? + return true if @user().connection().status() isnt z.user.ConnectionStatus.ACCEPTED and not @user().is_me + return false + , @, deferEvaluation: true + + @connection_is_not_established = ko.computed => + @user()?.connection().status() in [z.user.ConnectionStatus.PENDING, z.user.ConnectionStatus.SENT, z.user.ConnectionStatus.IGNORED] + , @, deferEvaluation: true + + @user_is_removed_from_conversation = ko.computed => + return true if not @user()? or not @conversation()? + return not (@user() in @conversation().participating_user_ets()) + , @, deferEvaluation: true + + @render_common_contacts = ko.pureComputed => + return @user()?.id and not @user().connected() and not @user().is_me + + # footer + @get_footer_template = ko.computed => + return 'user-profile-footer-empty' if not @user()? + + ConversationType = z.conversation.ConversationType + status = @user().connection().status() + is_me = @user().is_me + + # When used in conversation! + if @conversation? + type = @conversation().type() + + if type in [ConversationType.ONE2ONE, ConversationType.CONNECT] + return 'user-profile-footer-profile' if is_me + return 'user-profile-footer-add-block' if status is z.user.ConnectionStatus.ACCEPTED + return 'user-profile-footer-pending' if status is z.user.ConnectionStatus.SENT + + else if type is ConversationType.REGULAR + return 'user-profile-footer-profile-leave' if is_me + return 'user-profile-footer-connect-remove' if status in [z.user.ConnectionStatus.UNKNOWN, z.user.ConnectionStatus.CANCELLED] + return 'user-profile-footer-pending-remove' if status in [z.user.ConnectionStatus.PENDING, z.user.ConnectionStatus.SENT, z.user.ConnectionStatus.IGNORED] + return 'user-profile-footer-message-remove' if status is z.user.ConnectionStatus.ACCEPTED + return 'user-profile-footer-unblock-remove' if status is z.user.ConnectionStatus.BLOCKED + + # When used in Search! + else + return 'user-profile-footer-unblock' if status is z.user.ConnectionStatus.BLOCKED + return 'user-profile-footer-pending' if status is z.user.ConnectionStatus.SENT + return 'user-profile-footer-ignore-accept' if status in [z.user.ConnectionStatus.PENDING, z.user.ConnectionStatus.IGNORED] + return 'user-profile-footer-add' if status in [z.user.ConnectionStatus.UNKNOWN, z.user.ConnectionStatus.CANCELLED] + + return 'user-profile-footer-empty' + + click_on_device: (client_et) => + @selected_device client_et + + click_on_device_detail_back_button: => + @selected_device null + + click_on_my_fingerprint_button: => + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-my-fingerprint' + data: + device: @client_repository.current_client + fingerprint_local: @fingerprint_local + click_on_show_my_devices: -> + amplify.publish z.event.WebApp.PROFILE.SETTINGS.SHOW + + click_on_reset_session: => + reset_progress = => + window.setTimeout => + @is_resetting_session false + , 550 + + @is_resetting_session true + @conversation_repository.reset_session @user().id, @selected_device().id, @conversation().id + .then -> reset_progress() + .catch -> reset_progress() + + click_on_verify_client: => + toggle_verified = !!!@selected_device().meta.is_verified() + + @client_repository.update_client_in_db @user().id, @selected_device().id, { + meta: + is_verified: toggle_verified + } + .then => @selected_device().meta.is_verified toggle_verified + .catch (error) => @logger.log @logger.levels.WARN, "Client cannot be updated: #{error.message}" + + on_tab_index_changed: (index) => + if index is 1 + + user_id = @user().id + @client_repository.get_clients_by_user_id user_id + .then (client_ets) => + if client_ets?.length > 0 + @user().devices client_ets + @devices_found true + else + @devices_found false + .catch (error) => + @logger.log @logger.levels.ERROR, "Unable to retrieve clients data for user '#{user_id}': #{error}" + +ko.components.register 'user-profile', + viewModel: createViewModel: (params, component_info) -> + return new z.components.UserProfileViewModel params, component_info + template: + element: 'user-profile-template' diff --git a/app/script/config.coffee b/app/script/config.coffee new file mode 100644 index 00000000000..f664ffe9f62 --- /dev/null +++ b/app/script/config.coffee @@ -0,0 +1,112 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} + + +z.config = + BROWSER_NOTIFICATION: + TIMEOUT: 5000 + TITLE_LENGTH: 38 + BODY_LENGTH: 80 + + LOGGER: + OPTIONS: + name_length: 60 + domains: + 'app.wire.com': -> 0 + 'localhost': -> 300 + 'wire.ms': -> 300 + 'wire-webapp-edge.wire.com': -> 300 + 'wire-webapp-staging.wire.com': -> 300 + 'zinfra.io': -> 300 + + TIME_BETWEEN_PING: 30000 + + # number of message that will be pulled + MESSAGES_FETCH_LIMIT: 30 + + # number of users displayed in people you may know + SUGGESTIONS_FETCH_LIMIT: 30 + + # number of top people displayed in the start ui + TOP_PEOPLE_FETCH_LIMIT: 24 + + #Accent color IDs + ACCENT_ID: + BLUE: 1 + GREEN: 2 + YELLOW: 3 + RED: 4 + ORANGE: 5 + PINK: 6 + PURPLE: 7 + + # Ignored by the emoji lib + EXCLUDE_EMOJI: [ + '\u2122' # trademark + '\u00A9' # copyright + '\u00AE' # registered + ] + + # Conversation size + MAXIMUM_CONVERSATION_SIZE: 128 + + # self profile image size + MINIMUM_PROFILE_IMAGE_SIZE: + WIDTH: 320 + HEIGHT: 320 + + # 5 megabyte image upload limit + MAXIMUM_IMAGE_FILE_SIZE: 5 * 1024 * 1024 + + # 25 megabyte upload limit ( minus iv and padding ) + MAXIMUM_ASSET_FILE_SIZE: 25 * 1024 * 1024 - 16 - 16 + + # Maximum of parallel uploads + MAXIMUM_ASSET_UPLOADS: 10 + + # Maximum characters per message + MAXIMUM_MESSAGE_LENGTH: 8000 + + SUPPORTED_IMAGE_TYPES: [ + 'image/jpg', + 'image/jpeg', + 'image/png', + 'image/bmp' + ] + + MINIMUM_USERNAME_LENGTH: 2 + MINIMUM_PASSWORD_LENGTH: 8 + + # Time until phone code expires + LOGIN_CODE_EXPIRATION: 10 * 60 + + # measured in pixel + SCROLL_TO_LAST_MESSAGE_THRESHOLD: 100 + + # defines if it was a recently viewed conversation (5 min) + CONVERSATION_ACTIVITY_TIMEOUT: 5 * 60 * 1000 + + PROPERTIES_KEY: 'webapp' + + # bigger requests will be split in chunks with a maximum size as defined + MAXIMUM_USERS_PER_REQUEST: 200 + + UNSPLASH_URL: 'https://source.unsplash.com/1200x1200/?landscape' + ANNOUNCE_URL: 'https://wire.com/api/v1/announce/' diff --git a/app/script/connect/ConnectError.coffee b/app/script/connect/ConnectError.coffee new file mode 100644 index 00000000000..04f7153b343 --- /dev/null +++ b/app/script/connect/ConnectError.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +class z.connect.ConnectError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @stack = (new Error()).stack + @type = type + + @:: = new Error() + @::constructor = @ + @::TYPE = + GOOGLE_CLIENT: 'z.connect.ConnectError::TYPE.GOOGLE_CLIENT' + GOOGLE_DOWNLOAD: 'z.connect.ConnectError::TYPE.GOOGLE_DOWNLOAD' + NO_CONTACTS: 'z.connect.ConnectError::TYPE.NO_CONTACTS' + UPLOAD: 'z.connect.ConnectError::TYPE.UPLOAD' diff --git a/app/script/connect/ConnectGoogleService.coffee b/app/script/connect/ConnectGoogleService.coffee new file mode 100644 index 00000000000..739de94db7d --- /dev/null +++ b/app/script/connect/ConnectGoogleService.coffee @@ -0,0 +1,129 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +### +Connect Google Service for calls to the Google's REST API. + +@see https://github.com/google/google-api-javascript-client + https://developers.google.com/api-client-library/javascript/ + https://developers.google.com/google-apps/contacts/v3 + Use updated-min for newer updates + max-results +### +class z.connect.ConnectGoogleService + # Construct a new Connect Google Service. + constructor: (@client) -> + @logger = new z.util.Logger 'z.connect.ConnectGoogleService', z.config.LOGGER.OPTIONS + @client_id = '481053726221-71f8tbhghg4ug5put5v3j5pluv0di2fc.apps.googleusercontent.com' + @scopes = 'https://www.googleapis.com/auth/contacts.readonly' + + ### + Retrieves the user's Google Contacts. + @return [Promise] Promise that resolves with the Google contacts + ### + get_contacts: => + @_init_library() + .then => + return @_get_access_token() + .then (access_token) => + @_get_contacts access_token + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to import contacts from Google: #{error.message}", error + + ### + Authenticate before getting the contacts. + @private + @return [Promise] Promise that resolves when the user has been successfully authenticated + ### + _authenticate: => + return new Promise (resolve, reject) => + @logger.log @logger.levels.INFO, 'Authenticating with Google for contacts access' + + on_response = (response) => + if not response?.error + @logger.log @logger.levels.INFO, 'Received access token from Google', response + resolve response.access_token + else + @logger.log @logger.levels.ERROR, 'Failed to authenticate with Google', response + reject response?.error + + window.gapi.auth.authorize {client_id: @client_id, scope: @scopes, immediate: false}, on_response + + ### + Check for cached access token or authenticate with Google. + @return [Promise] Promise that resolves with the access token + ### + _get_access_token: => + return new Promise (resolve, reject) => + if window.gapi.auth + if auth_token = window.gapi.auth.getToken() + @logger.log @logger.levels.INFO, 'Using cached access token to access Google contacts', auth_token + resolve auth_token.access_token + else + @_authenticate().then(resolve).catch reject + else + error_message = 'Google Auth Client for JavaScript not loaded' + @logger.log @logger.levels.WARN, error_message + error = new z.connect.ConnectError error_message, z.connect.ConnectError::TYPE.GOOGLE_CLIENT + Raygun.send error + reject error + + ### + Retrieve the user's Google Contacts using a call to their backend. + @private + @return [Promise] Promise that resolves with the user's contacts + ### + _get_contacts: (access_token) -> + return new Promise (resolve, reject) => + @logger.log @logger.levels.INFO, 'Fetching address book from Google' + api_endpoint = 'https://www.google.com/m8/feeds/contacts/default/full' + $.get "#{api_endpoint}?access_token=#{access_token}&alt=json&max-results=15000&v=3.0" + .always (data_or_jqXHR, textStatus) => + if textStatus isnt 'error' + @logger.log @logger.levels.INFO, 'Received address book from Google', data_or_jqXHR + resolve data_or_jqXHR + else + @logger.log @logger.levels.ERROR, 'Failed to fetch address book from Google', data_or_jqXHR + reject data_or_jqXHR.responseJSON or new z.service.BackendClientError data_or_jqXHR.status + + ### + Initialize Google Auth Client for JavaScript is loaded. + @return [Promise] Promise that resolves when the authentication library is initialized + ### + _init_library: -> + if window.gapi + return Promise.resolve() + else + return @_load_library() + + ### + Lazy loading of the Google Auth Client for JavaScript. + @return [Promise] Promise that resolves when the authentication library is loaded + ### + _load_library: -> + return new Promise (resolve) => + window.gapi_loaded = resolve + + @logger.log @logger.levels.INFO, 'Lazy loading Google Auth API' + script_node = document.createElement 'script' + script_node.src = 'https://apis.google.com/js/auth.js?onload=gapi_loaded' + script_element = document.getElementsByTagName('script')[0] + script_element.parentNode.insertBefore script_node, script_element diff --git a/app/script/connect/ConnectRepository.coffee b/app/script/connect/ConnectRepository.coffee new file mode 100644 index 00000000000..b392f4c9e8c --- /dev/null +++ b/app/script/connect/ConnectRepository.coffee @@ -0,0 +1,196 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +# Connect Repository for all address book interactions with the connect service. +class z.connect.ConnectRepository + ### + Construct a new Connect Repository. + + @param connect_service [z.connect.ConnectService] Backend REST API service implementation + @param connect_google_service [z.connect.ConnectGoogleService] Google REST API implementation + @param user_repository [z.user.UserRepository] Repository for all user and connection interactions + ### + constructor: (@connect_service, @connect_google_service, @user_repository) -> + @logger = new z.util.Logger 'z.connect.ConnectRepository', z.config.LOGGER.OPTIONS + + ### + Retrieve a user's Google Contacts. + @return [Promise] Promise that resolves with the user's Google contacts that match on Wire + ### + get_google_contacts: -> + return new Promise (resolve, reject) => + @connect_google_service.get_contacts() + .catch (error) => + error_message = 'Google Contacts SDK error' + @logger.log @logger.levels.INFO, error_message, error + connect_error = new z.connect.ConnectError error_message, z.connect.ConnectError::TYPE.GOOGLE_DOWNLOAD + reject connect_error + throw connect_error + .then (response) => + return @_parse_google_contacts response + .then (phone_book) => + if phone_book.cards.length is 0 + error_message = 'No contacts found for upload' + @logger.log @logger.levels.WARN, error_message + resolve [] + throw new z.connect.ConnectError error_message, z.connect.ConnectError::TYPE.NO_CONTACTS + else + @logger.log @logger.levels.INFO, "Uploading hashes of '#{phone_book.cards.length}' contacts for matching", phone_book + return @connect_service.post_onboarding phone_book + .then (response) => + @logger.log @logger.levels.INFO, + "Gmail contacts upload successful: #{response.results.length} matches, #{response['auto-connects'].length} auto connects", response + @user_repository.save_property_contact_import_google Date.now() + resolve response + .catch (error) => + if error not instanceof z.connect.ConnectError + if error.code is z.service.BackendClientError::STATUS_CODE.TOO_MANY_REQUESTS + error_message = 'Backend refused Gmail contacts upload: Endpoint used too frequent' + @logger.log @logger.levels.ERROR, error_message + else + error_message = 'Gmail contacts upload failed' + @logger.log @logger.levels.ERROR, error_message, error + reject new z.connect.ConnectError error_message, z.connect.ConnectError::TYPE.UPLOAD + + + ### + Retrieve a user's OSX address book contacts. + @return [Promise] Promise that resolves with the user's address book contacts that match on Wire + ### + get_osx_contacts: -> + return new Promise (resolve, reject) => + phone_book = @_parse_osx_contacts() + + if phone_book.cards.length is 0 + error_message = 'No contacts found for upload' + @logger.log @logger.levels.WARN, error_message + reject new z.connect.ConnectError 'No contacts found for upload', z.connect.ConnectError::TYPE.NO_CONTACTS + else + @logger.log @logger.levels.INFO, "Uploading hashes of '#{phone_book.cards.length}' contacts for matching", phone_book + @connect_service.post_onboarding phone_book + .then (response) => + @logger.log @logger.levels.INFO, + "OS X contacts upload successful: #{response.results.length} matches, #{response['auto-connects'].length} auto connects", response + @user_repository.save_property_contact_import_osx Date.now() + resolve response + .catch (error) => + if error.code is z.service.BackendClientError::STATUS_CODE.TOO_MANY_REQUESTS + error_message = 'Backend refused OS X contacts upload: Endpoint used too frequent' + @logger.log @logger.levels.ERROR, error_message + else + error_message = 'OS X contacts upload failed' + @logger.log @logger.levels.ERROR, error_message, error + reject new z.connect.ConnectError error_message, z.connect.ConnectError::TYPE.UPLOAD + + ### + Encode phone book + + @private + @param phone_book [z.connect.PhoneBook] Object containing un-encoded phone book data + @return [z.connect.PhoneBook] Object containing encoded phone book data + ### + _encode_phone_book: (phone_book) -> + for entry, index in phone_book.self + phone_book.self[index] = z.util.encode_sha256_base64 entry + + for card, card_index in phone_book.cards + for contact, contact_index in card.contact + card.contact[contact_index] = z.util.encode_sha256_base64 contact + phone_book.cards[card_index] = card + + return phone_book + + ### + Parse a user's OSX address book Contacts. + @private + @return [z.connect.PhoneBook] Encoded phone book data + ### + _parse_osx_contacts: -> + return if not window.zAddressBook + + if _.isFunction window.zAddressBook + address_book = window.zAddressBook() # for osx >= 2.7 + else + address_book = window.zAddressBook # for osx < 2.7 + + phone_book = new z.connect.PhoneBook @user_repository.self() + + me = address_book.getMe() + for email in me.emails + phone_book.self.push email + for number in me.numbers + phone_book.self.push number + + x = 0 + while x < address_book.contactCount() + contact = address_book.getContact x + card = + contact: [] + card_id: CryptoJS.MD5("#{contact.firstName}#{contact.lastName}").toString() + for email in contact.emails + card.contact.push email.toLowerCase().trim() + for number in contact.numbers + card.contact.push z.util.phone_number_to_e164 number, navigator.language + + if card.contact.length > 0 + phone_book.cards.push card + x++ + + return @_encode_phone_book phone_book + + ### + Parse a user's Google Contacts. + @private + @param response [JSON] Response from Google API + @return [z.connect.PhoneBook] Encoded phone book data + ### + _parse_google_contacts: (response) -> + phone_book = new z.connect.PhoneBook @user_repository.self() + + # Add self info from Google + if response.feed.author? + self = response.feed.author + google_email = self[0].email.$t.toLowerCase().trim() + if not @user_repository.self().email() is google_email + phone_book.self.push google_email + + # Add Google contacts + if response.feed.entry? + users = response.feed.entry + for user in users + if user.gd$email? or user.gd$phoneNumber? + card = + contact: [] + card_id: user.gd$etag + + if user.gd$email? + for email in user.gd$email + card.contact.push email.address.toLowerCase().trim() + + if user.gd$phoneNumber? + for number in user.gd$phoneNumber + if number.uri? + card.contact.push number.uri + else + card.contact.push number.$t + + phone_book.cards.push card + return @_encode_phone_book phone_book diff --git a/app/script/connect/ConnectService.coffee b/app/script/connect/ConnectService.coffee new file mode 100644 index 00000000000..57a932f237d --- /dev/null +++ b/app/script/connect/ConnectService.coffee @@ -0,0 +1,42 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +# Connect Service for all addressbook calls to the backend REST API. +class z.connect.ConnectService + ### + Construct a new Connect Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.connect.ConnectService', z.config.LOGGER.OPTIONS + + ### + Upload address book data for matching. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/addressbook/onboardingV3 + + @param phone_book [Object] Phone book containing the address cards + @return [Promise] Promise that resolves with the matched contacts from the user's phone book + ### + post_onboarding: (phone_book) -> + @client.send_json + type: 'POST' + url: @client.create_url '/onboarding/v3' + data: phone_book diff --git a/app/script/connect/ConnectSource.coffee b/app/script/connect/ConnectSource.coffee new file mode 100644 index 00000000000..1b6dd2d39fd --- /dev/null +++ b/app/script/connect/ConnectSource.coffee @@ -0,0 +1,24 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +z.connect.ConnectSource = + GMAIL: 'gmail' + ICLOUD: 'icloud' diff --git a/app/script/connect/ConnectTrigger.coffee b/app/script/connect/ConnectTrigger.coffee new file mode 100644 index 00000000000..7ff0d8f43f8 --- /dev/null +++ b/app/script/connect/ConnectTrigger.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +z.connect.ConnectTrigger = + ONBOARDING: 'registration' + SEARCH: 'search' + SETTINGS: 'settings' diff --git a/app/script/connect/PhoneBook.coffee b/app/script/connect/PhoneBook.coffee new file mode 100644 index 00000000000..243afcd0c1b --- /dev/null +++ b/app/script/connect/PhoneBook.coffee @@ -0,0 +1,30 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.connect ?= {} + +# Phone book entity. +class z.connect.PhoneBook + ### + Construct a new Phone book. + @param self_user [z.entity.User] Self user + ### + constructor: (self_user) -> + @self = [self_user.email()] + @cards = [] diff --git a/app/script/conversation/ConversationMapper.coffee b/app/script/conversation/ConversationMapper.coffee new file mode 100644 index 00000000000..f1cccdac44b --- /dev/null +++ b/app/script/conversation/ConversationMapper.coffee @@ -0,0 +1,156 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Conversation Mapper to convert all server side JSON conversation objects into core entities +class z.conversation.ConversationMapper + # Construct a new Conversation Mapper. + constructor: -> + @logger = new z.util.Logger 'z.conversation.ConversationMapper', z.config.LOGGER.OPTIONS + + ### + Convert a JSON conversation into a conversation entity. + @param json [Object] Conversation data + @return [z.entity.Conversation] Mapped conversation entity + ### + map_conversation: (json) -> + conversation_ets = @map_conversations [json?.data or json] + return conversation_ets[0] + + ### + Convert multiple JSON conversations into a conversation entities. + @param json [Object] Conversation data + @return [Array] Mapped conversation entities + ### + map_conversations: (json) -> + return (@_create_conversation_et conversation for conversation in json) + + ### + Updates all properties of a conversation specified + + @example data: {"name":"ThisIsMyNewConversationName"} + @todo make utility? + + @param conversation_et [z.entity.Conversation] Conversation to be updated + @param data [Object] Conversation data + @return [z.entity.Conversation] Updated conversation entity + ### + update_properties: (conversation_et, data) -> + for key, value of data + continue if key is 'id' + if conversation_et[key]? + if ko.isObservable conversation_et[key] + conversation_et[key] value + else + conversation_et[key] = value + + return conversation_et + + ### + Update the membership properties of a conversation. + + @param conversation_et [z.entity.Conversation] Conversation to be updated + @param self [Object] Conversation self data + @return [z.entity.Conversation] Updated conversation entity + ### + update_self_status: (conversation_et, self) -> + return if not conversation_et? + + if self.status? + conversation_et.removed_from_conversation self.status is z.conversation.ConversationStatus.PAST_MEMBER + + # Last Event Timestamp from storage + if self.last_event_timestamp + conversation_et.set_timestamp self.last_event_timestamp, + z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + + if self.otr_archived? + timestamp = new Date(self.otr_archived_ref).getTime() + conversation_et.set_timestamp timestamp, z.conversation.ConversationUpdateType.ARCHIVED_TIMESTAMP + conversation_et.archived_state self.otr_archived + + if self.archived_timestamp + timestamp = self.archived_timestamp + conversation_et.set_timestamp timestamp, z.conversation.ConversationUpdateType.ARCHIVED_TIMESTAMP + conversation_et.archived_state self.archived_state + + if self.cleared_timestamp + conversation_et.set_timestamp self.cleared_timestamp, z.conversation.ConversationUpdateType.CLEARED_TIMESTAMP + + # Last read + if self.last_read_timestamp + conversation_et.set_timestamp self.last_read_timestamp, z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + + # Muted + if self.otr_muted? + timestamp = new Date(self.otr_muted_ref).getTime() + conversation_et.set_timestamp timestamp, z.conversation.ConversationUpdateType.MUTED_TIMESTAMP + conversation_et.muted_state self.otr_muted + + if self.muted_timestamp + conversation_et.set_timestamp self.muted_timestamp, z.conversation.ConversationUpdateType.MUTED_TIMESTAMP + conversation_et.muted_state self.muted_state + + return conversation_et + + ### + Creates a conversation entity from JSON data. + + @private + @param data [Object] Conversation data + @return [z.entity.Conversation] Mapped conversation entity + ### + _create_conversation_et: (data) -> + return if not data? + return @_update_conversation_et new z.entity.Conversation(data.id), data + + ### + Updates a given conversation entity from JSON data. + + @private + @param conversation_et [z.entity.Conversation] Conversation to be updated + @param data [Object] Conversation data + @return [z.entity.Conversation] Updated conversation entity + ### + _update_conversation_et: (conversation_et, data) -> + self = data.members.self + others = data.members.others + + conversation_et.id = data.id + conversation_et.creator = data.creator + conversation_et.type data.type + conversation_et.name data.name ? '' + + # Last event + timestamp = new Date(data.last_event_time).getTime() + conversation_et.set_timestamp timestamp, z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + + conversation_et = @update_self_status conversation_et, self + + # all users ( with all status codes ) + conversation_et.all_user_ids others.map (value) -> value.id + + # all users that are still active + participating_user_ids = [] + others.forEach (other) -> + participating_user_ids.push other.id if other.status is z.conversation.ConversationStatus.CURRENT_MEMBER + conversation_et.participating_user_ids participating_user_ids + + return conversation_et diff --git a/app/script/conversation/ConversationRepository.coffee b/app/script/conversation/ConversationRepository.coffee new file mode 100644 index 00000000000..ba887b1b98e --- /dev/null +++ b/app/script/conversation/ConversationRepository.coffee @@ -0,0 +1,1813 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Conversation repository for all conversation interactions with the conversation service +class z.conversation.ConversationRepository + ### + Construct a new Conversation Repository. + + @param conversation_service [z.conversation.ConversationService] Backend REST API conversation service implementation + @param asset_service [z.assets.AssetService] Backend REST API asset service implementation + @param user_repository [z.user.UserRepository] Repository for all user and connection interactions + @param giphy_repository [z.extension.GiphyRepository] Repository for Giphy GIFs + @param cryptography_repository [z.cryptography.CryptographyRepository] Repository for all cryptography interactions + @param link_repository [z.links.LinkPreviewRepository] Repository for link previews + ### + constructor: (@conversation_service, @asset_service, @user_repository, @giphy_repository, @cryptography_repository, @link_repository) -> + @logger = new z.util.Logger 'z.conversation.ConversationRepository', z.config.LOGGER.OPTIONS + + @conversation_mapper = new z.conversation.ConversationMapper() + @event_mapper = new z.conversation.EventMapper @asset_service, @user_repository + + @active_conversation = ko.observable() + + @conversations = ko.observableArray [] + @has_initialized_participants = false + @is_handling_notifications = true + @fetching_conversations = {} + + @self_conversation = ko.computed => + return @find_conversation_by_id @user_repository.self().id if @user_repository.self() + + @filtered_conversations = ko.computed => + @conversations().filter (conversation_et) -> + states_to_filter = [z.user.ConnectionStatus.BLOCKED, z.user.ConnectionStatus.CANCELLED, z.user.ConnectionStatus.PENDING] + return false if conversation_et.connection().status() in states_to_filter + return false if conversation_et.is_self() + return false if conversation_et.is_cleared() and conversation_et.removed_from_conversation() + return true + + @sorted_conversations = ko.computed => + @filtered_conversations().sort z.util.sort_groups_by_last_event + + @conversations_archived = ko.observableArray [] + @conversations_call = ko.observableArray [] + @conversations_cleared = ko.observableArray [] + @conversations_unarchived = ko.observableArray [] + + @processed_event_ids = {} + @processed_event_nonces = {} + + @_init_subscriptions() + + _init_state_updates: -> + ko.computed => + archived = [] + calls = [] + cleared = [] + unarchived = [] + + for conversation_et in @sorted_conversations() + if conversation_et.has_active_call() + calls.push conversation_et + else if conversation_et.is_cleared() + cleared.push conversation_et + else if conversation_et.is_archived() + archived.push conversation_et + else + unarchived.push conversation_et + + @conversations_archived archived + @conversations_call calls + @conversations_cleared cleared + @conversations_unarchived unarchived + + _init_subscriptions: -> + amplify.subscribe z.event.WebApp.CONVERSATION.ASSET.CANCEL, @cancel_asset_upload + amplify.subscribe z.event.WebApp.CONVERSATION.EVENT_FROM_BACKEND, @on_conversation_event + amplify.subscribe z.event.WebApp.CONVERSATION.MAP_CONNECTION, @map_connection + amplify.subscribe z.event.WebApp.CONVERSATION.STORE, @save_conversation_in_db + amplify.subscribe z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, @set_notification_handling_state + amplify.subscribe z.event.WebApp.SELF.CLIENT_ADD, @on_self_client_add + amplify.subscribe z.event.WebApp.USER.UNBLOCKED, @unblocked_user + amplify.subscribe z.event.WebApp.CONVERSATION.MESSAGE.DELETE, @delete_message + + + ############################################################################### + # Conversation service interactions + ############################################################################### + + ### + Create a new conversation. + @note Supply at least 2 user IDs! Do not include the requestor + + @param user_ids [Array] IDs of users (excluding the requestor) to be part of the conversation + @param name [String] User defined name for the Conversation (optional) + @param on_success [Function] Function to be called on success + @param on_error [Function] Function to be called on failure + ### + create_new_conversation: (user_ids, name, on_success, on_error) => + @conversation_service.create_conversation user_ids, name, (response, error) => + if response + on_success? @create response + else + on_error? error + + ### + Get a conversation from the backend. + @param conversation_et [z.entity.Conversation] Conversation to be saved + @return [Boolean] Is the conversation active + ### + fetch_conversation_by_id: (conversation_id, callback) -> + for id, callbacks of @fetching_conversations when id is conversation_id + callbacks.push callback + return + + @fetching_conversations[conversation_id] = [callback] + + @conversation_service.get_conversation_by_id conversation_id, (response, error) => + if response + conversation_et = @conversation_mapper.map_conversation response + @save_conversation conversation_et + @logger.log @logger.levels.INFO, "Conversation with ID '#{conversation_id}' fetched from backend" + callbacks = @fetching_conversations[conversation_id] + for callback in callbacks + callback? conversation_et + delete @fetching_conversations[conversation_id] + else + @logger.log @logger.levels.ERROR, "Conversation with ID '#{conversation_id}' could not be fetched from backend" + + ### + Retrieve all conversations using paging. + + @param limit [Integer] Query limit for conversation + @param conversation_id [String] ID of the last conversation in batch + @return [Promise] Promise that resolves when all conversations have been retrieved and saved + ### + get_conversations: (limit = 500, conversation_id) => + return new Promise (resolve, reject) => + @conversation_service.get_conversations limit, conversation_id + .then (response) => + if response.has_more + last_conversation_et = response.conversations[response.conversations.length - 1] + @get_conversations limit, last_conversation_et.id + .then => resolve @conversations() + + if response.conversations.length > 0 + conversation_ets = @conversation_mapper.map_conversations response.conversations + @save_conversations conversation_ets + + if not response?.has_more + @load_conversation_states() + resolve @conversations() + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to retrieve conversations from backend: #{error.message}", error + reject error + + ### + Get conversation events. + @param conversation_et [z.entity.Conversation] Conversation to start from + ### + get_events: (conversation_et) -> + conversation_et.is_pending true + timestamp = conversation_et.get_first_message()?.timestamp + @conversation_service.load_events_from_db conversation_et.id, timestamp + .then (loaded_events) => + [events, has_further_events] = loaded_events + conversation_et.has_further_messages has_further_events + if events.length is 0 + @logger.log @logger.levels.INFO, "No events for conversation '#{conversation_et.id}' found", events + else if timestamp + date = new Date(timestamp).toISOString() + @logger.log @logger.levels.INFO, + "Loaded #{events.length} event(s) starting at '#{date}' for conversation '#{conversation_et.id}'", events + else + @logger.log @logger.levels.INFO, + "Loaded first #{events.length} event(s) for conversation '#{conversation_et.id}'", events + raw_events = (event.mapped or event.raw for event in events) + @_add_events_to_conversation events: raw_events, conversation_et + conversation_et.is_pending false + .catch (error) => + @logger.log @logger.levels.INFO, "Could not load events for conversation: #{conversation_et.id}", error + + ### + Get conversation unread events. + @param conversation_et [z.entity.Conversation] Conversation to start from + ### + _get_unread_events: (conversation_et) -> + conversation_et.is_pending true + timestamp = conversation_et.get_first_message()?.timestamp + @conversation_service.load_unread_events_from_db conversation_et, timestamp + .then (events) => + if events.length + raw_events = (event.mapped or event.raw for event in events) + @_add_events_to_conversation events: raw_events, conversation_et + conversation_et.is_pending false + .catch (error) => + @logger.log @logger.levels.INFO, "Could not load unread events for conversation: #{conversation_et.id}", error + + ### + Update conversation with a user you just unblocked + @param user_et [z.entity.User] User you unblocked + ### + unblocked_user: (user_et) => + conversation_et = @get_one_to_one_conversation user_et.id + conversation_et?.removed_from_conversation false + + ### + Get users and events for conversations. + @note To reduce the number of backend calls we merge the user IDs of all conversations first. + @param conversation_ets [Array] Array of conversation entities to be updated + ### + update_conversations: (conversation_ets) => + user_ids = _.flatten(conversation_et.all_user_ids() for conversation_et in conversation_ets) + @user_repository.get_users_by_id user_ids, => + @_fetch_users_and_events conversation_et for conversation_et in conversation_ets + + ### + Map users to conversations without any backend requests. + @param conversation_ets [Array] Array of conversation entities to be updated + ### + update_conversations_offline: (conversation_ets) => + @update_participating_user_ets conversation_et, undefined, true for conversation_et in conversation_ets + + + ############################################################################### + # Repository interactions + ############################################################################### + + ### + Find a local conversation by ID. + @param conversation_id [String] ID of conversation to get + @return [z.entity.Conversation] Conversation + ### + find_conversation_by_id: (conversation_id) -> + return conversation for conversation in @conversations() when conversation.id is conversation_id + + get_all_users_in_conversation: (conversation_id) -> + return new Promise (resolve) => + @get_conversation_by_id conversation_id, (conversation_et) => + others = conversation_et.participating_user_ets() + resolve others.concat [@user_repository.self()] + + ### + Check for conversation locally and fetch it from the server otherwise. + @param conversation_id [String] ID of conversation to get + @param callback [Function] Function to be called on server return + ### + get_conversation_by_id: (conversation_id, callback) -> + if not conversation_id + Raygun.send new Error 'Trying to get conversation without ID' + return + + conversation_et = @find_conversation_by_id conversation_id + if callback + if conversation_et? + callback? conversation_et + else + @fetch_conversation_by_id conversation_id, callback + + return conversation_et + + ### + Get group conversations by name + @param group_name [String] Query to be searched in group conversation names + @return [Array] Matching group conversations + ### + get_groups_by_name: (group_name) => + @sorted_conversations().filter (conversation_et) -> + return false if not conversation_et.is_group() + return true if z.util.compare_names conversation_et.display_name(), group_name + for user_et in conversation_et.participating_user_ets() + return true if z.util.compare_names user_et.name(), group_name + return false + + ### + Get the next unarchived conversation. + @param conversation_et [z.entity.Conversation] Conversation to start from + @return [z.entity.Conversation] Next conversation + ### + get_next_conversation: (conversation_et) -> + return z.util.array_get_next @conversations_unarchived(), conversation_et + + ### + Get unarchived conversation with the most recent event. + @return [z.entity.Conversation] Most recent conversation + ### + get_most_recent_conversation: -> + return @conversations_unarchived()?[0] + + ### + Get conversation with a user. + @param user_id [String] ID of user for whom to get the conversation + @return [z.entity.Conversation] Conversation with requested user + ### + get_one_to_one_conversation: (user_id) => + for conversation_et in @conversations() + if conversation_et.type() in [z.conversation.ConversationType.ONE2ONE, z.conversation.ConversationType.CONNECT] + return conversation_et if user_id is conversation_et.participating_user_ids()[0] + + ### + Check whether conversation is currently displayed. + @param conversation_et [z.entity.Conversation] Conversation to be saved + @return [Boolean] Is the conversation active + ### + is_active_conversation: (conversation_et) -> + return @active_conversation() is conversation_et + + ### + Check whether message has been read. + + @param conversation_id [String] Conversation ID + @param message_id [String] Message ID + @return [Boolean] Is the message marked as read + ### + is_message_read: (conversation_id, message_id) => + conversation_et = @get_conversation_by_id conversation_id + message_et = conversation_et.get_message_by_id message_id + + if not message_et + @logger.log @logger.levels.WARN, "Message ID '#{message_id}' not found for conversation ID '#{conversation_id}'" + return true + + return conversation_et.last_read_timestamp() >= message_et.timestamp + + ### + Load the conversation states from the store. + ### + load_conversation_states: => + @conversation_service.load_conversation_states_from_db() + .then (conversation_states) => + for state in conversation_states + conversation_et = @get_conversation_by_id state.id + @conversation_mapper.update_self_status conversation_et, state + + @logger.log @logger.levels.INFO, "Updated '#{conversation_states.length}' conversation states" + amplify.publish z.event.WebApp.CONVERSATION.LOADED_STATES + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to update conversation states', error + + ### + Maps a connection to the corresponding conversation + + @note If there is no conversation it will request it from the backend + @param [Array] Connections + @param [Boolean] open the new conversation + ### + map_connection: (connection_ets, show_conversation = false) => + for connection_et in connection_ets + conversation_et = @get_conversation_by_id connection_et.conversation_id + + # We either accepted a pending connection request or send new connection request + states_to_fetch = [z.user.ConnectionStatus.ACCEPTED, z.user.ConnectionStatus.SENT] + if not conversation_et and connection_et.status() in states_to_fetch + @fetch_conversation_by_id connection_et.conversation_id, (conversation_et) => + if conversation_et + @save_conversation conversation_et + conversation_et.connection connection_et + @update_participating_user_ets conversation_et, (conversation_et) -> + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et if show_conversation + else if conversation_et? + conversation_et.connection connection_et + @update_participating_user_ets conversation_et, (conversation_et) -> + if connection_et.status() is z.user.ConnectionStatus.ACCEPTED + conversation_et.type z.conversation.ConversationType.ONE2ONE + + if not @has_initialized_participants + @logger.log @logger.levels.INFO, 'Updating group participants offline' + @_init_state_updates() + @update_conversations_offline @conversations_unarchived() + @update_conversations_offline @conversations_archived() + @update_conversations_offline @conversations_cleared() + @has_initialized_participants = true + + ### + Mark conversation as read. + @param conversation_et [z.entity.Conversation] Conversation to be marked as read + ### + mark_as_read: (conversation_et) => + return if conversation_et is undefined + return if @is_handling_notifications + return if conversation_et.number_of_unread_events() is 0 + return if conversation_et.get_last_message()?.type is z.event.Backend.CONVERSATION.MEMBER_UPDATE + + @_update_last_read_timestamp conversation_et + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.REMOVE_READ + + ### + Save a conversation in the repository. + @param conversation_et [z.entity.Conversation] Conversation to be saved in the repository + ### + save_conversation: (conversation_et) => + if not @get_conversation_by_id conversation_et.id + @conversations.push conversation_et + @save_conversation_in_db conversation_et + + save_conversation_in_db: (conversation_et, updated_field) => + @conversation_service.save_conversation_in_db conversation_et, updated_field + + ### + Save conversations in the repository. + @param conversation_ets [Array] Conversations to be saved in the repository + ### + save_conversations: (conversation_ets) => + z.util.ko_array_push_all @conversations, conversation_ets + + ### + Set the notification handling state. + @note Temporarily do not unarchive conversations when handling the notification stream + @param handling_notifications [Boolean] Notifications are being handled from the stream + ### + set_notification_handling_state: (handling_notifications) => + @is_handling_notifications = handling_notifications + @logger.log @logger.levels.INFO, "Ignoring events for unarchiving: #{handling_notifications}" + + ### + Update participating users in a conversation. + + @param conversation_et [z.entity.Conversation] Conversation to be updated + @param callback [Function] Function to be called on server return + @param offline [Boolean] Should we only look for cached contacts + ### + update_participating_user_ets: (conversation_et, callback, offline = false) => + conversation_et.self = @user_repository.self() + user_ids = conversation_et.participating_user_ids() + @user_repository.get_users_by_id user_ids, (user_ets) -> + conversation_et.participating_user_ets.removeAll() + z.util.ko_array_push_all conversation_et.participating_user_ets, user_ets + callback? conversation_et + , offline + + + ############################################################################### + # Send events + ############################################################################### + + ### + Add users to an existing conversation. + + @param conversation_et [z.entity.Conversation] Conversation to add users to + @param user_ids [Array] IDs of users to be added to the conversation + @param callback [Function] Function to be called on server return + ### + add_members: (conversation_et, users_ids, callback) => + @conversation_service.post_members conversation_et.id, users_ids + .then (response) -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.SessionEventName.INTEGER.USERS_ADDED_TO_CONVERSATIONS, users_ids.length + amplify.publish z.event.WebApp.EVENT.INJECT, response + callback?() + .catch (error_response) -> + if error_response.label is z.service.BackendClientError::LABEL.TOO_MANY_MEMBERS + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.TOO_MANY_MEMBERS, + data: + max: z.config.MAXIMUM_CONVERSATION_SIZE + open_spots: Math.max 0, z.config.MAXIMUM_CONVERSATION_SIZE - (conversation_et.number_of_participants() + 1) + + ### + Archive a conversation. + @param conversation_et [z.entity.Conversation] Conversation to rename + @param next_conversation_et [z.entity.Conversation] Next conversation to potentially switch to + ### + archive_conversation: (conversation_et, next_conversation_et) => + + # other clients just use the old event id as a flag. if archived is + # set they consider the conversation is archived + payload = + otr_archived: true + otr_archived_ref: new Date(conversation_et.last_event_timestamp()).toISOString() + + @conversation_service.update_member_properties conversation_et.id, payload + .then => + @member_update conversation_et, {data: payload}, next_conversation_et + @logger.log @logger.levels.INFO, + "Archived conversation '#{conversation_et.id}' on '#{payload.otr_archived_ref}'" + .catch (error) => + @logger.log @logger.levels.ERROR, + "Conversation '#{conversation_et.id}' could not be archived: #{error.code}\r\nPayload: #{JSON.stringify(payload)}", error + + ### + Clear conversation content and archive the conversation. + + @note According to spec we archive a conversation when we clear it. + It will be unarchived once it is opened through search. We use the archive flag to distinguish states. + + @param conversation_et [z.entity.Conversation] Conversation to clear + @param leave [Boolean] Should we leave the conversation before clearing the content? + ### + clear_conversation: (conversation_et, leave = false) => + next_conversation_et = @get_next_conversation conversation_et + + _clear_conversation = => + @_update_cleared_timestamp conversation_et + @_delete_messages conversation_et + amplify.publish z.event.WebApp.CONVERSATION.SHOW, next_conversation_et + + if leave + @leave_conversation conversation_et, next_conversation_et, _clear_conversation + else + _clear_conversation() + + _update_cleared_timestamp: (conversation_et) -> + cleared_timestamp = conversation_et.last_event_timestamp() + + if conversation_et.set_timestamp cleared_timestamp, z.conversation.ConversationUpdateType.CLEARED_TIMESTAMP + message_content = new z.proto.Cleared conversation_et.id, cleared_timestamp + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'cleared', message_content + + @_send_encrypted_value @self_conversation().id, generic_message + .then => + @logger.log @logger.levels.INFO, + "Cleared conversation '#{conversation_et.id}' as read on '#{new Date(cleared_timestamp).toISOString()}'" + .catch (error) => + @logger.log @logger.levels.ERROR, "Error (#{error.label}): #{error.message}" + error = new Error 'Event response is undefined' + Raygun.send error, source: 'Sending encrypted last read' + + ### + Leave conversation. + + @param conversation_et [z.entity.Conversation] Conversation to leave + @param next_conversation_et [z.entity.Conversation] Next conversation in list + @param callback [Function] Function to be called on server return + ### + leave_conversation: (conversation_et, next_conversation_et, callback) => + @conversation_service.delete_members conversation_et.id, @user_repository.self().id + .then (response) => + amplify.publish z.event.WebApp.EVENT.INJECT, response + @member_leave conversation_et, response + .then => + if callback? + callback next_conversation_et + else + @archive_conversation conversation_et, next_conversation_et + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to leave conversation (#{conversation_et.id}): #{error}" + + ### + Remove member from conversation. + + @param conversation_et [z.entity.Conversation] Conversation to remove member from + @param user_id [String] ID of member to be removed from the the conversation + @param callback [Function] Function to be called on server return + ### + remove_member: (conversation_et, user_id, callback) => + @conversation_service.delete_members conversation_et.id, user_id + .then (response) -> + amplify.publish z.event.WebApp.EVENT.INJECT, response + callback?() + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to remove member from conversation (#{conversation_et.id}): #{error}" + + ### + Rename conversation. + + @param conversation_et [z.entity.Conversation] Conversation to rename + @param name [String] New conversation name + @param callback [Function] Function to be called on server return + ### + rename_conversation: (conversation_et, name, callback) => + @conversation_service.update_conversation_properties conversation_et.id, name + .then (response) -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.CONVERSATION_RENAMED + amplify.publish z.event.WebApp.EVENT.INJECT, response + .then -> + callback?() + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to rename conversation (#{conversation_et.id}): #{error}" + + reset_session: (user_id, client_id, conversation_id) => + @logger.log @logger.levels.INFO, "Resetting session with client '#{client_id}' of user '#{user_id}'" + @cryptography_repository.reset_session user_id, client_id + .then (session_id) => + if session_id + @logger.log @logger.levels.INFO, "Deleted session with client '#{client_id}' of user '#{user_id}'" + else + @logger.log @logger.levels.WARN, 'No local session found to delete' + return @send_encrypted_session_reset user_id, client_id, conversation_id + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to reset session for client '#{client_id}' of user '#{user_id}': #{error.message}", error + throw error + + reset_all_sessions: => + sessions = @cryptography_repository.storage_repository.sessions + @logger.log @logger.levels.INFO, "Resetting '#{Object.keys(sessions).length}' sessions" + for session_id, session of sessions + ids = z.client.Client.dismantle_user_client_id session_id + if ids.user_id is @user_repository.self().id + conversation_et = @self_conversation() + else + conversation_et = @get_one_to_one_conversation ids.user_id + @reset_session ids.user_id, ids.client_id, conversation_et.id + + ### + Send a specific GIF to a conversation. + + @param conversation_et [z.entity.Conversation] Conversation to send message in + @param url [String] URL of giphy image + @param tag [String] tag tag used for gif search + @param callback [Function] Function to be called on server return + ### + send_gif: (conversation_et, url, tag, callback) => + if not tag + tag = z.localization.Localizer.get_text z.string.extensions_giphy_random + + message = z.localization.Localizer.get_text { + id: z.string.extensions_giphy_message + replace: {placeholder: '%tag', content: tag} + } + + z.util.load_url_blob url, (blob) => + @send_encrypted_message message, conversation_et + @upload_images conversation_et, [blob] + callback?() + + ### + Toggle a conversation between silence and notify. + @param conversation_et [z.entity.Conversation] Conversation to rename + ### + toggle_silence_conversation: (conversation_et) => + return new Promise (resolve, reject) => + if conversation_et.is_muted() + payload = + muted: false + otr_muted: false + otr_muted_ref: new Date().toISOString() + else + payload = + muted: true + muted_time: new Date().toJSON() + otr_muted: true + otr_muted_ref: new Date(conversation_et.last_event_timestamp()).toISOString() + + @conversation_service.update_member_properties conversation_et.id, payload + .then => + response = {data: payload} + @member_update conversation_et, response + @logger.log @logger.levels.INFO, + "Toggle silence to '#{payload.otr_muted}' for conversation '#{conversation_et.id}' on '#{payload.otr_muted_ref}'" + resolve response + .catch (error) => + reject_error = new Error "Conversation '#{conversation_et.id}' could not be muted: #{error.code}" + @logger.log @logger.levels.WARN, reject_error.message, error + reject reject_error + + ### + Un-archive a conversation. + @param conversation_et [z.entity.Conversation] Conversation to rename + @param callback [Function] Function to be called on return + ### + unarchive_conversation: (conversation_et, callback) => + return new Promise (resolve, reject) => + payload = + otr_archived: false + otr_archived_ref: new Date(conversation_et.last_event_timestamp()).toISOString() + + @conversation_service.update_member_properties conversation_et.id, payload + .then => + response = {data: payload} + @member_update conversation_et, response + @logger.log @logger.levels.INFO, + "Unarchived conversation '#{conversation_et.id}' on '#{payload.otr_archived_ref}'" + callback?() + resolve response + .catch (error) => + reject_error = new Error "Conversation '#{conversation_et.id}' could not be unarchived: #{error.code}" + @logger.log @logger.levels.WARN, reject_error.message, error + callback?() + reject reject_error + + ### + Update last read of conversation using timestamp. + @private + @param conversation_et [z.entity.Conversation] Conversation to update + ### + _update_last_read_timestamp: (conversation_et) -> + timestamp = conversation_et.get_last_message()?.timestamp + return if not timestamp? + + if conversation_et.set_timestamp timestamp, z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + message_content = new z.proto.LastRead conversation_et.id, conversation_et.last_read_timestamp() + + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'lastRead', message_content + + @_send_encrypted_value @self_conversation().id, generic_message + .then => + @logger.log @logger.levels.INFO, + "Marked conversation '#{conversation_et.id}' as read on '#{new Date(timestamp).toISOString()}'" + .catch (error) => + @logger.log @logger.levels.ERROR, "Error (#{error.label}): #{error.message}" + error = new Error 'Event response is undefined' + Raygun.send error, source: 'Sending encrypted last read' + + + ############################################################################### + # Send encrypted events + ############################################################################### + + ### + Send encrypted assets. Used for file transfers. + + # TODO unify with image asset + # create and send original proto message (message-add) + # create and send uploaded proto message with status (asset-add) + + @param conversation_id [String] Conversation ID + @return [Object] Collection with User IDs which hold their Client IDs in an Array + ### + send_encrypted_asset: (conversation_et, file, nonce) => + conversation_id = conversation_et.id + generic_message = null + key_bytes = null + sha256 = null + ciphertext = null + body_payload = null + initial_payload = null + + return Promise.resolve() + .then -> + message_et = conversation_et.get_message_by_id nonce + asset_et = message_et.assets()[0] + asset_et.upload_id nonce # TODO combine + asset_et.uploaded_on_this_client true + + return z.util.load_file_buffer file + .then (file_bytes) -> + return z.assets.AssetCrypto.encrypt_aes_asset file_bytes + .then (data) -> + # TODO send original message and + [key_bytes, sha256, ciphertext] = data + key_bytes = new Uint8Array key_bytes + sha256 = new Uint8Array sha256 + .then => + return @_create_user_client_map conversation_id + .then (user_client_map) => + generic_message = new z.proto.GenericMessage nonce + generic_message.set 'asset', @_construct_asset_uploaded key_bytes, sha256 + return @cryptography_repository.encrypt_generic_message user_client_map, generic_message + .then (payload) => + payload.inline = false + payload.native_push = true + body_payload = new Uint8Array ciphertext + initial_payload = payload + return @asset_service.post_asset_v2 conversation_id, payload, body_payload, false, nonce + .catch (error_response) => + return @_update_payload_for_changed_clients error_response, generic_message, initial_payload + .then (updated_payload) => + @asset_service.post_asset_v2 conversation_id, updated_payload, body_payload, true, nonce + .then (response) => + [json, asset_id] = response + event = @_construct_otr_asset_event response, conversation_et.id, asset_id + event.data.otr_key = key_bytes + event.data.sha256 = sha256 + event.id = nonce + return @asset_upload_complete conversation_et, event + + ### + When we reset a session then we must inform the remote client about this action. + + @param conversation_et [z.entity.Conversation] Conversation that should receive the file + @param file [File] File to send + ### + send_encrypted_asset_metadata: (conversation_et, file) => + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'asset', @_construct_asset_original file + @_send_and_save_encrypted_value conversation_et, generic_message + + ### + When we reset a session then we must inform the remote client about this action. + + @param conversation_et [z.entity.Conversation] Conversation that should receive the file + @param nonce [String] id of the metadata message + @param reason [z.assets.AssetUploadFailedReason] cause for the failed upload (optional) + ### + send_encrypted_asset_upload_failed: (conversation_et, nonce, reason = z.assets.AssetUploadFailedReason.FAILED) => + generic_message = new z.proto.GenericMessage nonce + generic_message.set 'asset', @_construct_asset_not_uploaded reason + @_send_and_save_encrypted_value conversation_et, generic_message + + ### + Send encrypted external message + + @param conversation_et [z.entity.Conversation] Conversation that should receive the message + @param generic_message [z.protobuf.GenericMessage] Generic message to be sent as external message + @return [Promise] Promise that resolves after sending the external message + ### + send_encrypted_external_message: (conversation_et, generic_message) => + @logger.log @logger.levels.INFO, "Sending external message of type '#{generic_message.content}'", generic_message + + conversation_id = conversation_et.id + key_bytes = null + sha256 = null + ciphertext = null + initial_payload = null + + z.assets.AssetCrypto.encrypt_aes_asset generic_message.toArrayBuffer() + .then (data) => + [key_bytes, sha256, ciphertext] = data + return @_create_user_client_map conversation_id + .then (user_client_map) => + generic_message_external = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message_external.set 'external', new z.proto.External new Uint8Array(key_bytes), new Uint8Array(sha256) + return @cryptography_repository.encrypt_generic_message user_client_map, generic_message_external + .then (payload) => + payload.data = z.util.array_to_base64 ciphertext + payload.native_push = true + initial_payload = payload + return @conversation_service.post_encrypted_message conversation_id, payload, false + .catch (error_response) => + return @_update_payload_for_changed_clients error_response, generic_message, initial_payload + .then (updated_payload) => + return @conversation_service.post_encrypted_message conversation_id, updated_payload, true + .then (response) => + event = @_construct_otr_message_event response, conversation_et.id + return @cryptography_repository.save_encrypted_event generic_message, event + .then (record) => + @add_event conversation_et, record.mapped if record?.mapped + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to send external message for conversation with id '#{conversation_id}'", error + + ### + Sends an OTR Image Asset + ### + send_encrypted_image_asset: (conversation_et, image) => + return new Promise (resolve, reject) => + asset = null + ciphertext = null + conversation_id = conversation_et.id + generic_message = null + initial_payload = null + + @asset_service.create_asset_proto image + .then ([asset, asset_ciphertext]) => + ciphertext = asset_ciphertext + generic_message = new z.proto.GenericMessage z.util.create_random_uuid(), null, asset + return @_create_user_client_map conversation_id + .then (user_client_map) => + return @cryptography_repository.encrypt_generic_message user_client_map, generic_message + .then (payload) => + initial_payload = payload + initial_payload.inline = false + initial_payload.native_push = true + return @asset_service.post_asset_v2 conversation_id, initial_payload, ciphertext, false + .catch (error_response) => + return @_update_payload_for_changed_clients error_response, generic_message, initial_payload + .then (updated_payload) => + @asset_service.post_asset_v2 conversation_id, updated_payload, ciphertext, true + .then ([json, asset_id]) => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.IMAGE_SENT + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.COMPLETED_MEDIA_ACTION, + action: 'photo', conversation_type: if conversation_et.is_one2one() then 'one_to_one' else 'group' + event = @_construct_otr_asset_event json, conversation_id, asset_id + return @cryptography_repository.save_encrypted_event generic_message, event + .then (record) => + @add_event conversation_et, record.mapped + resolve() + .catch (error) => + @logger.log "Failed to upload otr asset for conversation #{conversation_id}", error + exception = new Error('Event response is undefined') + custom_data = + source: 'Sending medium image' + error: error + Raygun.send exception, custom_data + reject error + + ### + Send an encrypted knock. + @param conversation_et [z.entity.Conversation] Conversation to send knock in + @return [Promise] Promise that resolves after sending the knock + ### + send_encrypted_knock: (conversation_et) => + @_send_and_save_encrypted_value conversation_et, new z.proto.Knock false + .then -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.PING_SENT + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.COMPLETED_MEDIA_ACTION, + action: 'ping', conversation_type: if conversation_et.is_one2one() then 'one_to_one' else 'group' + .catch (error) => @logger.log @logger.levels.ERROR, "#{error.message}" + + ### + Send message to specific converation. + + @note Will either send a normal or external message. + @param message [String] plain text message + @param conversation_et [z.entity.Conversation] Conversation that should receive the message + @return [Promise] Promise that resolves after sending the message + ### + send_encrypted_message_with_link_preview: (message, url, offset, conversation_et) => + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'text', new z.proto.Text message + + @_send_and_save_encrypted_value conversation_et, generic_message + .then => + @link_repository.get_link_preview url, offset + .then (link_preview) => + generic_message.text.link_preview.push link_preview + @_send_and_save_encrypted_value conversation_et, generic_message + .catch (error) => + @logger.log @logger.levels.ERROR, 'Error while sending link preview', error + + ### + Send message to specific converation. + + @note Will either send a normal or external message. + @param message [String] plain text message + @param conversation_et [z.entity.Conversation] Conversation that should receive the message + @param retry [Boolean] Try to resend as external with first attempt failed (optional) + @return [Promise] Promise that resolves after sending the message + ### + send_encrypted_message: (message, conversation_et, retry = true) => + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'text', new z.proto.Text message + + Promise.resolve() + .then => + if @_send_as_external_message conversation_et, generic_message + @send_encrypted_external_message conversation_et, generic_message + else + @_send_and_save_encrypted_value conversation_et, generic_message + .then (message_record) => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.MESSAGE_SENT + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.COMPLETED_MEDIA_ACTION, + action: 'text', conversation_type: if conversation_et.is_one2one() then 'one_to_one' else 'group' + @_analyze_sent_message message + return message_record + .catch (error) => + if error.code is z.service.BackendClientError::STATUS_CODE.REQUEST_TOO_LARGE and retry + @send_encrypted_external_message conversation_et, generic_message + else + @logger.log @logger.levels.ERROR, "#{error.message}", error + error = new Error "Failed to send message: #{error.message}" + custom_data = + source: 'Sending message' + Raygun.send error, custom_data + throw error + + ### + Sending a message to the remote end of a session reset. + + @note When we reset a session then we must inform the remote client about this action. It sends a ProtocolBuffer message + (which will not be rendered in the view) to the remote client. This message only needs to be sent to the affected + remote client, therefore we force the message sending. + + @param user_id [String] User ID + @param client_id [String] Client ID + @param conversation_id [String] Conversation ID + @return [Promise] Promise that resolves after sending the session reset + ### + send_encrypted_session_reset: (user_id, client_id, conversation_id) => + return new Promise (resolve, reject) => + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.setClientAction z.proto.ClientAction.RESET_SESSION + + user_client_map = @_create_user_client_map_from_ids user_id, client_id + + @cryptography_repository.encrypt_generic_message user_client_map, generic_message + .then (payload) => + return @conversation_service.post_encrypted_message conversation_id, payload, true + .then (response) => + @logger.log @logger.levels.INFO, "Sent info about session reset to client '#{client_id}' of user '#{user_id}'" + resolve response + .catch (error) => + @logger.log @logger.levels.ERROR, "Sending conversation reset failed: #{error.message}", error + reject error + + ### + Create not uploaded Asset protobuf message (failure). + + @private + @param reason [z.assets.AssetUploadFailedReason] Conversation ID + @return [z.proto.Asset] Asset protobuf message + ### + _construct_asset_not_uploaded: (reason) -> + asset = new z.proto.Asset() + if reason is z.assets.AssetUploadFailedReason.CANCELLED + asset.set 'not_uploaded', z.proto.Asset.NotUploaded.CANCELLED + else + asset.set 'not_uploaded', z.proto.Asset.NotUploaded.FAILED + return asset + + ### + Create original Asset protobuf message. + + @private + @param file [Object] File data + @return [z.proto.Asset] Asset protobuf message + ### + _construct_asset_original: (file) -> + original_asset = new z.proto.Asset.Original file.type, file.size, file.name + asset = new z.proto.Asset() + asset.set 'original', original_asset + return asset + + ### + Create uploaded Asset proto (success). + + @private + @param otr_key [ByteArray] Encryption key + @param sha256 [ByteArray] Sha256 + @return [z.proto.Asset] Asset protobuf message + ### + _construct_asset_uploaded: (otr_key, sha256) -> + uploaded_asset = new z.proto.Asset.RemoteData otr_key, sha256 + asset = new z.proto.Asset() + asset.set 'uploaded', uploaded_asset + return asset + + ### + Create MsgDeleted protobuf message. + + @private + @param conversation_id [String] Conversation ID + @param message_id [String] ID of message to be deleted + @return [z.proto.MsgDeleted] MsgDeleted protobuf message + ### + _construct_delete: (conversation_id, message_id) -> + return new z.proto.MsgDeleted conversation_id, message_id + + ### + Construct an encrypted message event. + + @private + @param response [JSON] Backend response + @param conversation_id [String] Conversation ID + @return [Object] Object in form of 'conversation.otr-message-add' + ### + _construct_otr_message_event: (response, conversation_id) -> + event = + data: undefined + from: @user_repository.self().id + time: response.time + type: 'conversation.otr-message-add' + conversation: conversation_id + + return event + + ### + Construct an encrypted asset event. + + @private + @param response [JSON] Backend response + @param conversation_id [String] Conversation ID + @param asset_id [String] Asset ID + @return [Object] Object in form of 'conversation.otr-asset-add' + ### + _construct_otr_asset_event: (response, conversation_id, asset_id) -> + event = + data: + id: asset_id + from: @user_repository.self().id + time: response.time + type: 'conversation.otr-asset-add' + conversation: conversation_id + + return event + + ### + Create a user client map for a given conversation. + + @private + @param conversation_id [String] Conversation ID + @return [Promise] Promise that resolves with a user client map + ### + _create_user_client_map: (conversation_id) -> + @get_all_users_in_conversation conversation_id + .then (user_ets) -> + user_client_map = {} + + for user_et in user_ets when user_et.devices()[0] + user_client_map[user_et.id] = (client_et.id for client_et in user_et.devices()) + + return user_client_map + + ### + Create a user client map for given IDs. + + @private + @param user_id [String] User ID + @param client_id [String] Client ID + @return [Object] User client map + ### + _create_user_client_map_from_ids: (user_id, client_id) -> + user_client_map = {} + user_client_map[user_id] = [client_id] + return user_client_map + + ### + Saves and sends a generic message to the conversation + + @private + @param conversation_et [z.entity.Conversation] Conversation to send message to + @param message_content [z.proto] Protobuf message content to be added to generic message + @return [Promise] Promise that resolves when message has been added to the conversation + ### + _send_and_save_encrypted_value: (conversation_et, message_content) => + return new Promise (resolve, reject) => + reject() if conversation_et.removed_from_conversation() + + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + + if message_content instanceof z.proto.Knock + generic_message.set 'knock', message_content + else if message_content instanceof z.proto.Text + generic_message.set 'text', message_content + else if message_content instanceof z.proto.GenericMessage + generic_message = message_content + + @_send_encrypted_value conversation_et.id, generic_message + .then (response) => + event = @_construct_otr_message_event response, conversation_et.id + return @cryptography_repository.save_encrypted_event generic_message, event + .then (record) => + @add_event conversation_et, record.mapped if record?.mapped + resolve record + .catch (error) => + error_message = "Could not send OTR message of type '#{generic_message.content}' to conversation ID '#{conversation_et.id}' (#{conversation_et.display_name()}): #{error.message}" + + raygun_error = new Error 'Encryption failed' + custom_data = error_message: "Could not send OTR message of type '#{generic_message.content}'" + Raygun.send raygun_error, custom_data + + @logger.log @logger.levels.ERROR, error_message, {error: error, event: generic_message} + if error.label is z.service.BackendClientError::LABEL.UNKNOWN_CLIENT + amplify.publish z.event.WebApp.SIGN_OUT, 'unknown_sender', true + reject error + + ### + Sends a generic message to the conversation + + @private + @param conversation_id [String] Conversation ID + @param generic_message [z.protobuf.GenericMessage] Protobuf message to be encrypted and send + @return [Promise] Promise that resolves after sending the encrypted message + ### + _send_encrypted_value: (conversation_id, generic_message) => + initial_payload = null + @_create_user_client_map conversation_id + .then (user_client_map) => + return @cryptography_repository.encrypt_generic_message user_client_map, generic_message + .then (payload) => + initial_payload = payload + @logger.log @logger.levels.INFO, "Sending encrypted message to conversation '#{conversation_id}'", payload + return @conversation_service.post_encrypted_message conversation_id, payload, false + .catch (error_response) => + return @_update_payload_for_changed_clients error_response, generic_message, initial_payload + .then (updated_payload) => + @logger.log @logger.levels.INFO, "Sending updated encrypted message to conversation '#{conversation_id}'", updated_payload + return @conversation_service.post_encrypted_message conversation_id, updated_payload, true + + ### + Estimate whether message should be send as type external. + + @private + @param conversation_et [z.entitity.Conversation] Conversation entity + @param generic_message [z.protobuf.GenericMessage] Generic message that will be send + @return [Boolean] Is payload likely to be too big so that we switch to type external? + ### + _send_as_external_message: (conversation_et, generic_message) -> + estimated_number_of_clients = conversation_et.number_of_participants() * 4 + message_in_bytes = new Uint8Array(generic_message.toArrayBuffer()).length + estimated_payload_in_bytes = estimated_number_of_clients * message_in_bytes + return estimated_payload_in_bytes / 1024 > 200 + + ### + Post images to a conversation. + + @param conversation_et [z.entity.Conversation] Conversation to post the images + @param images [Object] Message content + ### + upload_images: (conversation_et, images) => + return if not @_can_upload_assets_to_conversation conversation_et + @send_encrypted_image_asset conversation_et, image for image in images + + ### + Post files to a conversation. + @param conversation_et [z.entity.Conversation] Conversation to post the files + @param files [Object] File objects + ### + upload_files: (conversation_et, files) => + return if not @_can_upload_assets_to_conversation conversation_et + @upload_file conversation_et, file for file in files + + ### + Post file to a conversation. + + @param conversation_et [z.entity.Conversation] Conversation to post the file + @param file [Object] File object + ### + upload_file: (conversation_et, file) => + return if not @_can_upload_assets_to_conversation conversation_et + message_et = null + + upload_started = Date.now() + tracking_data = + size_bytes: file.size + size_mb: z.util.bucket_values (file.size / 1024 / 1024), [0, 5, 10, 15, 20, 25] + type: z.util.get_file_extension file.name + conversation_type = if conversation_et.is_one2one() then 'one_to_one' else 'group' + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.UPLOAD_INITIATED, + $.extend tracking_data, {conversation_type: conversation_type} + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.MEDIA.COMPLETED_MEDIA_ACTION, + action: 'file', conversation_type: conversation_type + + @send_encrypted_asset_metadata conversation_et, file + .then (record) => + message_et = conversation_et.get_message_by_id record.mapped.id + @send_encrypted_asset conversation_et, file, record.mapped.id + .then => + upload_duration = (Date.now() - upload_started) / 1000 + @logger.log "Finished to upload asset for conversation'#{conversation_et.id} in #{upload_duration}" + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.UPLOAD_SUCCESSFUL, + $.extend tracking_data, {time: upload_duration} + .catch (error) => + @logger.log "Failed to upload asset for conversation'#{conversation_et.id}", error + if message_et.id + @send_encrypted_asset_upload_failed conversation_et, message_et.id + @update_message_as_upload_failed message_et + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.UPLOAD_FAILED, tracking_data + + ### + Delete message in conversation. + + @param conversation_et [z.entity.Conversation] Conversation to post the file + @param message_et [z.entity.Message] Message object + ### + delete_message: (message_et) => + conversation_et = @active_conversation() + if message_et? + generic_message = new z.proto.GenericMessage z.util.create_random_uuid() + generic_message.set 'deleted', @_construct_delete conversation_et.id, message_et.id + + @_send_encrypted_value @self_conversation().id, generic_message + .then => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.DELETED_MESSAGE, {mode: 'single'} + return @_delete_message conversation_et, message_et.id + .catch (error) -> + @logger.log "Failed to send delete message with id '#{message_id}' for conversation '#{conversation_et.id}'", error + + ### + Can user upload assets to conversation. + + # TODO move to conversation et + + @param conversation_et [z.entity.Conversation] Conversation to rename + ### + _can_upload_assets_to_conversation: (conversation_et) -> + return false if not conversation_et? + return false if conversation_et.removed_from_conversation() + return false if conversation_et.is_request() + return false if conversation_et.is_one2one() and conversation_et.connection().status() isnt z.user.ConnectionStatus.ACCEPTED + return true + + + ############################################################################### + # Event callbacks + ############################################################################### + + message_deleted: (event_json) => + conversation_id = event_json.data.conversation_id + message_id = event_json.data.message_id + + if conversation_id? and message_id? + conversation_et = @find_conversation_by_id conversation_id + @_delete_message conversation_et, message_id + else + @logger.log "Failed to delete message with id '#{message_id}'' for conversation '#{conversation_et.id}'" + + ### + A message or ping received in a conversation. + @param conversation_et [z.entity.Conversation] Conversation to add the event to + @param event_json [Object] JSON data of 'conversation.message-add' or 'conversation.knock' event + ### + add_event: (conversation_et, event_json) => + @_add_event_to_conversation event_json, conversation_et, (message_et) => + @_send_event_notification event_json, conversation_et, message_et + + ### + An asset was uploaded + @param event_json [Object] JSON data of 'conversation.asset-upload-complete' event + @return [z.entity.Conversation] The conversation that was created + ### + asset_upload_complete: (conversation_et, event_json) -> + message_et = conversation_et.get_message_by_id event_json.id + + if not message_et? + return @logger.log @logger.levels.ERROR, "Upload complete: Could not find message with id '#{event_json.id}'", event_json + + @update_message_as_upload_complete conversation_et, message_et, event_json.data + + ### + An asset failed + @param event_json [Object] JSON data of 'conversation.asset-upload-failed' event + @return [z.entity.Conversation] The conversation that was created + ### + asset_upload_failed: (conversation_et, event_json) -> + message_et = conversation_et.get_message_by_id event_json.id + + if not message_et? + return @logger.log @logger.levels.ERROR, "Upload failed: Could not find message with id '#{event_json.id}'", event_json + + if event_json.data.reason is z.assets.AssetUploadFailedReason.CANCELLED + @_delete_message conversation_et, message_et.id + else + @update_message_as_upload_failed message_et + + ### + An asset preview was send + @param event_json [Object] JSON data of 'conversation.asset-upload-failed' event + @return [z.entity.Conversation] The conversation that was created + ### + asset_preview: (conversation_et, event_json) -> + message_et = conversation_et.get_message_by_id event_json.id + + if not message_et? + return @logger.log @logger.levels.ERROR, 'Asset preview: Could not find message with id '#{event_json.id}'", event_json + + @update_message_with_asset_preview conversation_et, message_et, event_json.data + + ### + A conversation was created. + @param event_json [Object] JSON data of 'conversation.create' event + @return [z.entity.Conversation] The conversation that was created + ### + create: (event_json) => + conversation_et = @find_conversation_by_id event_json.id + + if not conversation_et? + conversation_et = @conversation_mapper.map_conversation event_json + @update_participating_user_ets conversation_et, (conversation_et) => + @_send_conversation_create_notification conversation_et + @save_conversation conversation_et + + return conversation_et + + ### + User were added to a group conversation. + @param conversation_et [z.entity.Conversation] Conversation to add users to + @param event_json [Object] JSON data of 'conversation.member-join' event + ### + member_join: (conversation_et, event_json) => + for user_id in event_json.data.user_ids when user_id isnt @user_repository.self().id + conversation_et.participating_user_ids.push user_id if user_id not in conversation_et.participating_user_ids() + + # Self user joins again + if @user_repository.self().id in event_json.data.user_ids + conversation_et.removed_from_conversation false + + @update_participating_user_ets conversation_et, => + @_add_event_to_conversation event_json, conversation_et, (message_et) -> + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, conversation_et, message_et + + ### + Members of a group conversation were removed or left. + @param conversation_et [z.entity.Conversation] Conversation to remove users from + @param event_json [Object] JSON data of 'conversation.member-leave' event + ### + member_leave: (conversation_et, event_json) => + @_add_event_to_conversation event_json, conversation_et, (message_et) => + for user_et in message_et.user_ets() + if conversation_et.call() + if user_et.is_me + amplify.publish z.event.WebApp.CALL.STATE.DELETE, conversation_et.id + else + amplify.publish z.event.WebApp.CALL.STATE.REMOVE_PARTICIPANT, conversation_et.id, user_et.id + conversation_et.participating_user_ids.remove user_et.id + continue if not user_et.is_me + + conversation_et.removed_from_conversation true + if conversation_et.call() + amplify.publish z.event.WebApp.CALL.STATE.LEAVE, conversation_et.id + + @update_participating_user_ets conversation_et, -> + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, conversation_et, message_et + + ### + Membership properties for a conversation were updated. + + @param conversation_et [z.entity.Conversation] Conversation entity that will be updated + @param event_json [Object] JSON data of 'conversation.member-update' event + @param conversation_et [z.entity.Conversation] Next conversation in list + ### + member_update: (conversation_et, event_json, next_conversation_et) => + previously_archived = conversation_et.is_archived() + next_conversation_et = @get_next_conversation conversation_et if not next_conversation_et? + + @conversation_mapper.update_self_status conversation_et, event_json.data + + if previously_archived and not conversation_et.is_archived() + @_fetch_users_and_events conversation_et + else if conversation_et.is_archived() + amplify.publish z.event.WebApp.CONVERSATION.SWITCH, conversation_et, next_conversation_et + + ### + A conversation was renamed. + @param conversation_et [z.entity.Conversation] Conversation entity that will be renamed + @param event_json [Object] JSON data of 'conversation.rename' event + ### + rename: (conversation_et, event_json) => + @_add_event_to_conversation event_json, conversation_et, (message_et) => + @conversation_mapper.update_properties conversation_et, event_json.data + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, conversation_et, message_et + + + ############################################################################### + # Private + ############################################################################### + + ### + Convert a JSON event into an entity and add it to a given conversation. + + @param json [Object] Event data + @param conversation_et [z.entity.Conversation] Conversation entity the event will be added to + @param callback [Function] Function to be called on return + ### + _add_event_to_conversation: (json, conversation_et, callback) -> + message_et = @event_mapper.map_json_event json, conversation_et + @_update_user_ets message_et, (message_et) => + if conversation_et + conversation_et.add_message message_et + else + @logger.log @logger.levels.ERROR, + "Message cannot be added to unloaded conversation. Message type: #{message_et.type}" + error = new Error 'Conversation not loaded, message cannot be added' + custom_data = + message_type: message_et.type + Raygun.send error, custom_data + callback? message_et + + ### + Convert multiple JSON events into entities and add them to a given conversation. + + @param json [Object] Event data + @param conversation_et [z.entity.Conversation] Conversation entity the events will be added to + @param prepend [Boolean] Should existing messages be prepended + ### + _add_events_to_conversation: (json, conversation_et, prepend = true) -> + return if not json? + + message_ets = @event_mapper.map_json_events json, conversation_et + for message_et in message_ets + @_update_user_ets message_et + if prepend and conversation_et.messages().length > 0 + conversation_et.prepend_messages message_ets + else + conversation_et.add_messages message_ets + + ### + Check for duplicates by event IDs and cache the event ID. + + @private + @param message_et [z.entity.Message] Message entity + @param conversation_et [z.entity.Conversation] Conversation entity + @return [Boolean] Returns true if event is a duplicate + ### + _check_for_duplicate_event_by_nonce: (message_et, conversation_et) -> + return false if not message_et.nonce + event_nonce = "#{conversation_et.id}:#{message_et.nonce}:#{message_et.assets?()[0].type or message_et.super_type}" + if @processed_event_nonces[event_nonce] is undefined + @processed_event_nonces[event_nonce] = null + # @todo Maybe we need to reset "@processed_event_nonces" someday to save some memory, until now it's fine. + return false + else + @logger.log @logger.levels.WARN, "Event with nonce has been already processed : #{event_nonce}", message_et + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.SessionEventName.INTEGER.EVENT_HIDDEN_DUE_TO_DUPLICATE_NONCE + return true + + ### + Fetch all unread events and users of a conversation. + @private + @param conversation_et [z.entity.Conversation] Conversation fetch events and users for + ### + _fetch_users_and_events: (conversation_et) -> + if not conversation_et.is_loaded() and not conversation_et.is_pending() + @update_participating_user_ets conversation_et + @_get_unread_events conversation_et + + ### + @example Look for ongoing call + has_call = @_has_active_event response.events, [z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE], [z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE] + ### + _has_active_event: (events, include_on, exclude_on) -> + event_is_active = false + + for event in events + if event.type in include_on + event_is_active = true + else if event.type in exclude_on + event_is_active = false + + return event_is_active + + ### + Forward the 'conversation.create' event to the SystemNotification repository for browser and audio notifications. + @private + @param conversation_et [z.entity.Conversation] Conversation that was created + ### + _send_conversation_create_notification: (conversation_et) -> + @user_repository.get_user_by_id conversation_et.creator, (user_et) -> + message_et = new z.entity.MemberMessage() + message_et.user user_et + message_et.member_message_type = z.message.SystemMessageType.CONVERSATION_CREATE + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, conversation_et, message_et + + ### + Forward the event to the SystemNotification repository for browser and audio notifications. + + @private + @param event_json [Object] JSON data of received event + @param conversation_et [z.entity.Conversation] Conversation that was created + @param message_et [z.entity.Message] Message that has been received + ### + _send_event_notification: (event_json, conversation_et, message_et) -> + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, conversation_et, message_et + + ### + Listener for incoming events from the WebSocket. + + @private + @note We check for events received multiple times via the WebSocket by event id here + @param event [Object] JSON data for event + #### + on_conversation_event: (event) => + if not event + error = new Error('Event response is undefined') + custom_data = + source: 'WebSocket' + Raygun.send error, custom_data + + @logger.log "»» Event: '#{event.type}'", {event_object: event, event_json: JSON.stringify event} + + # Ignore member join if we join a one2one conversation (accept a connection request) + if event.type is z.event.Backend.CONVERSATION.MEMBER_JOIN + connection_et = @user_repository.get_connection_by_conversation_id event.conversation + return if connection_et?.status() is z.user.ConnectionStatus.PENDING + + # Check if conversation was archived + @get_conversation_by_id event.conversation, (conversation_et) => + previously_archived = conversation_et.is_archived() + + switch event.type + when z.event.Backend.CONVERSATION.CREATE + @create event + when z.event.Backend.CONVERSATION.MEMBER_JOIN + @member_join conversation_et, event + when z.event.Backend.CONVERSATION.MEMBER_LEAVE + @member_leave conversation_et, event + when z.event.Backend.CONVERSATION.MEMBER_UPDATE + @member_update conversation_et, event + when z.event.Backend.CONVERSATION.RENAME + @rename conversation_et, event + when z.event.Backend.CONVERSATION.ASSET_UPLOAD_COMPLETE + @asset_upload_complete conversation_et, event + when z.event.Backend.CONVERSATION.ASSET_UPLOAD_FAILED + @asset_upload_failed conversation_et, event + when z.event.Backend.CONVERSATION.ASSET_PREVIEW + @asset_preview conversation_et, event + when z.event.Backend.CONVERSATION.MESSAGE_DELETE + @message_deleted event + else + @add_event conversation_et, event + + # Un-archive it also on the backend side + if not @is_handling_notifications and previously_archived and not conversation_et.is_archived() + @logger.log @logger.levels.INFO, "Unarchiving conversation '#{conversation_et.id}' with new event" + @unarchive_conversation conversation_et + + ### + Updates the user entities that are part of a message. + @param message_et [z.entity.Message] Message to be updated + @param callback [Function] Function to be called on return + ### + _update_user_ets: (message_et, callback) => + @user_repository.get_user_by_id message_et.from, (user_et) => + message_et.user user_et + if message_et.is_member() + @user_repository.get_users_by_id message_et.user_ids(), (user_ets) -> + message_et.user_ets user_ets + else if message_et.has_asset_text() + for asset_et in message_et.assets() when asset_et.is_text() + if not message_et.user() + Raygun.send new Error 'Message does not contain user when updating' + else + asset_et.theme_color = message_et.user().accent_color() + callback? message_et + + ### + Cancel asset upload. + @param message_et [z.entity.Message] message_et on which the cancel was initiated + ### + cancel_asset_upload: (message_et) => + conversation_et = @active_conversation() + @asset_service.cancel_asset_upload message_et.assets()[0].upload_id() + @_delete_message conversation_et, message_et.id + @send_encrypted_asset_upload_failed conversation_et, message_et.id, z.assets.AssetUploadFailedReason.CANCELLED + + _handle_deleted_clients: (deleted_client_map, payload) -> + return Promise.resolve() + .then => + if _.isEmpty deleted_client_map + @logger.log @logger.levels.INFO, 'No obsolete clients that need to be removed' + return payload + else + @logger.log @logger.levels.INFO, 'Removing payload for deleted clients', deleted_client_map + delete_promises = [] + for user_id, client_ids of deleted_client_map + for client_id in client_ids + @logger.log @logger.levels.WARN, "The client '#{client_id}' from '#{user_id}' is obsolete and will be removed" + delete payload.recipients[user_id][client_id] + delete_promises.push @user_repository.client_repository.delete_client_and_session user_id, client_id + delete payload.recipients[user_id] if Object.keys(payload.recipients[user_id]).length is 0 + + Promise.all delete_promises + .then -> + return payload + + _handle_missing_clients: (missing_client_map, generic_message, payload) -> + return Promise.resolve() + .then => + if _.isEmpty missing_client_map + @logger.log @logger.levels.INFO, 'No missing clients that need to be added' + return payload + else + @logger.log @logger.levels.INFO, "Adding payload for missing clients of '#{Object.keys(missing_client_map).length}' users", missing_client_map + save_promises = [] + + @cryptography_repository.encrypt_generic_message missing_client_map, generic_message, payload + .then (updated_payload) => + payload = updated_payload + for user_id, client_ids of missing_client_map + for client_id in client_ids + save_promises.push @user_repository.add_client_to_user user_id, new z.client.Client {id: client_id} + + return Promise.all save_promises + .then -> + return payload + + _update_payload_for_changed_clients: (error_response, generic_message, payload) => + return Promise.resolve() + .then => + if error_response.missing + @logger.log @logger.levels.WARN, 'Payload for clients was missing', error_response + @_handle_deleted_clients error_response.deleted, payload + .then (updated_payload) => + return @_handle_missing_clients error_response.missing, generic_message, updated_payload + else + throw error_response + + ### + Delete message from UI an database + @param conversation_et [z.entity.Conversation] Conversation that contains the message + @param message_id [String] Message to delete + ### + _delete_message: (conversation_et, message_id) => + conversation_et.remove_message_by_id message_id + @conversation_service.delete_message_from_db conversation_et.id, message_id + + ### + Delete messages from UI an database + @param conversation_et [z.entity.Conversation] Conversation that contains the message + ### + _delete_messages: (conversation_et) -> + conversation_et.remove_messages() + @conversation_service.delete_messages_from_db conversation_et.id + + ### + Update asset in UI and DB as failed + @param message_et [z.entity.Message] Message to update + ### + update_message_as_upload_failed: (message_et) => + asset_et = message_et.get_first_asset() + asset_et.status z.assets.AssetTransferState.UPLOAD_FAILED + asset_et.upload_failed_reason z.assets.AssetUploadFailedReason.FAILED + @conversation_service.update_asset_as_failed_in_db message_et.primary_key + + ### + Update asset in UI and DB as completed + + @param conversation_et [z.entity.Conversation] Conversation that contains the message + @param message_et [z.entity.Message] Message to delete + @param asset_data [Object] + @option id [Number] asset id + @option otr_key [Uint8Array] aes key + @option sha256 [Uint8Array] hash of the encrypted asset + ### + update_message_as_upload_complete: (conversation_et, message_et, asset_data) => + resource = z.assets.AssetRemoteData.v2 conversation_et.id, asset_data.id, asset_data.otr_key, asset_data.sha256 + asset_et = message_et.get_first_asset() + asset_et.original_resource resource + asset_et.status z.assets.AssetTransferState.UPLOADED + @conversation_service.update_asset_as_uploaded_in_db message_et.primary_key, asset_data + + ### + Update asset in UI and DB with preview + + @param conversation_et [z.entity.Conversation] Conversation that contains the message + @param message_et [z.entity.Message] Message to delete + @param asset_data [Object] + @option id [Number] asset id + @option otr_key [Uint8Array] aes key + @option sha256 [Uint8Array] hash of the encrypted asset + ### + update_message_with_asset_preview: (conversation_et, message_et, asset_data) => + resource = z.assets.AssetRemoteData.v2 conversation_et.id, asset_data.id, asset_data.otr_key, asset_data.sha256 + asset_et = message_et.get_first_asset() + asset_et.preview_resource resource + @conversation_service.update_asset_preview_in_db message_et.primary_key, asset_data + + + ############################################################################### + # Helpers + ############################################################################### + + ### + Archive all conversations but not the self conversation. + @note Archiving the self conversation will lead to problems on other clients (like Android). + ### + archive_all_conversations: => + @archive_conversation conversation_et for conversation_et in @conversations() when not conversation_et.is_self() + + ### + Clear and leave all conversations but not the self conversation. + ### + clear_all_conversations: => + @clear_conversation conversation_et, conversation_et.is_group() for conversation_et in @conversations_unarchived() + + ### + Un-archive all conversations (and even the self conversation). + @note Un-archiving all conversations can help to reset the client to a proper state. + ### + unarchive_all_conversations: => + @unarchive_conversation conversation_et for conversation_et in @conversations() + + + ############################################################################### + # Tracking helpers + ############################################################################### + + ### + Count of group conversations + @return [Integer] Number of group conversations + ### + get_number_of_group_conversations: -> + group_conversations = (i for conversation_et, i in @conversations() when conversation_et.is_group()) + return group_conversations.length + + ### + Count of silenced conversations + @return [Integer] Number of conversations that are silenced + ### + get_number_of_silenced_conversations: => + silenced_conversations = (i for conversation_et, i in @conversations() when conversation_et.is_muted()) + return silenced_conversations.length + + ### + Count number of pending uploads + @return [Integer] Number of pending uploads + ### + get_number_of_pending_uploads: => + return @conversations().reduce (sum, conversation_et) -> + sum + conversation_et.get_number_of_pending_uploads() + , 0 + + ### + Analyze sent text message for rich media content. + @private + @param message [String] Message content to be checked for rich media + ### + _analyze_sent_message: (message) -> + soundcloud_links = message.match z.media.MediaEmbeds.regex.soundcloud + if soundcloud_links + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.SessionEventName.INTEGER.SOUNDCLOUD_LINKS_SENT, soundcloud_links.length + + youtube_links = message.match z.media.MediaEmbeds.regex.youtube + if youtube_links + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.SessionEventName.INTEGER.YOUTUBE_LINKS_SENT, youtube_links.length + + ### + Analyze sent text message for rich media content. + @param client [Object] + ### + on_self_client_add: (client) => + return + self = @user_repository.self() + message_et = new z.entity.E2EEDeviceMessage() + message_et.user self + message_et.device client + message_et.device_owner self + + # TODO save message + for conversation_et in @filtered_conversations() + if conversation_et.type() in [z.conversation.ConversationType.ONE2ONE, z.conversation.ConversationType.REGULAR] + conversation_et.add_message message_et diff --git a/app/script/conversation/ConversationService.coffee b/app/script/conversation/ConversationService.coffee new file mode 100644 index 00000000000..8bbe3df6de1 --- /dev/null +++ b/app/script/conversation/ConversationService.coffee @@ -0,0 +1,420 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Conversation service for all conversation calls to the backend REST API. +class z.conversation.ConversationService + URL_CONVERSATIONS: '/conversations' + ### + Construct a new Conversation Service. + + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client, @storage_service) -> + @logger = new z.util.Logger 'z.conversation.ConversationService', z.config.LOGGER.OPTIONS + + ### + Saves a conversation entity in the local database. + + @param conversation_et [z.entity.Conversation] Conversation entity + @return [Promise] Promise that will resolve with the primary key of the persisted conversation entity + ### + _save_conversation_in_db: (conversation_et) -> + return new Promise (resolve, reject) => + store_name = @storage_service.OBJECT_STORE_CONVERSATIONS + @storage_service.save store_name, conversation_et.id, conversation_et.serialize() + .then (primary_key) => + @logger.log @logger.levels.INFO, "Conversation '#{primary_key}' was stored for the first time" + resolve() + .catch (error) => + @logger.log @logger.levels.ERROR, "Conversation '#{conversation_et.id}' could not be stored", error + reject error + + ### + Updates a conversation entity in the database. + + @param updated_field [z.conversation.ConversationUpdateType] Property of the conversation entity which needs to be updated in the local database + @return [Promise] Promise which resolves with the conversation entity (if update was successful) or the conversation entity (if it was a new entity) + ### + _update_conversation_in_db: (conversation_et, updated_field) -> + return new Promise (resolve, reject) => + store_name = @storage_service.OBJECT_STORE_CONVERSATIONS + + switch updated_field + when z.conversation.ConversationUpdateType.ARCHIVED_STATE + entity = + archived_state: conversation_et.archived_state() + archived_timestamp: conversation_et.archived_timestamp() + when z.conversation.ConversationUpdateType.CLEARED_TIMESTAMP + entity = cleared_timestamp: conversation_et.cleared_timestamp() + when z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + entity = last_event_timestamp: conversation_et.last_event_timestamp() + when z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + entity = last_read_timestamp: conversation_et.last_read_timestamp() + when z.conversation.ConversationUpdateType.MUTED_STATE + entity = + muted_state: conversation_et.muted_state() + muted_timestamp: conversation_et.muted_timestamp() + + @storage_service.update store_name, conversation_et.id, entity + .then (number_of_updated_records) => + if number_of_updated_records + @logger.log @logger.levels.INFO, + "Conversation '#{conversation_et.id}' got a persistent update for property '#{updated_field}'" + resolve conversation_et + else + @_save_conversation_in_db conversation_et + .catch (error) => + @logger.log @logger.levels.ERROR, "Conversation '#{conversation_et.id}' could not be updated", error + reject error + + ############################################################################### + # Create conversations + ############################################################################### + + ### + Create a new conversation. + + @note Supply at least 2 user IDs! Do not include the requestor + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/createGroupConversation + + @param user_ids [Array] IDs of users (excluding the requestor) to be part of the conversation + @param name [String] User defined name for the Conversation (optional) + @param callback [Function] Function to be called on server return + ### + create_conversation: (user_ids, name, callback) -> + @client.send_json + url: @client.create_url z.conversation.ConversationService::URL_CONVERSATIONS + type: 'POST' + data: + users: user_ids + name: name + callback: callback + + ### + Create a One:One conversation. + + @note Do not include the requestor + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/createOne2OneConversation + + @param user_ids [Array] IDs of users (excluding the requestor) to be part of the conversation + @param name [String] User defined name for the Conversation (optional) + @param callback [Function] Function to be called on server return + ### + create_one_to_one_conversation: (user_ids, name, callback) -> + @client.send_json + url: @client.create_url '/conversations/one2one' + type: 'POST' + data: + users: user_ids + name: name + callback: callback + + ############################################################################### + # Get conversations + ############################################################################### + + ### + Retrieves meta information about all the conversations of a user. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/conversations + + @param limit [Integer] Number of results to return (default 100, max 100) + @param conversation_id [String] Conversation ID to start from + ### + get_conversations: (limit = 100, conversation_id = undefined) -> + @client.send_request + url: @client.create_url z.conversation.ConversationService::URL_CONVERSATIONS + type: 'GET' + data: + size: limit + start: conversation_id + + ### + Get a conversation by ID. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/conversation + + @param conversation_id [String] ID of conversation to get + @param callback [Function] Function to be called on server return + ### + get_conversation_by_id: (conversation_id, callback) -> + @client.send_request + url: @client.create_url "/conversations/#{conversation_id}" + type: 'GET' + callback: callback + + ### + Get the last (i.e. most current) event ID per conversation. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/lastEvents + @todo Implement paging for this endpoint + + @param callback [Function] Function to be called on server return + ### + get_last_events: (callback) -> + @client.send_request + url: @client.create_url '/conversations/last-events' + type: 'GET' + callback: callback + + ############################################################################### + # Send events + ############################################################################### + + ### + Remove member from conversation. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/removeMember + + @param conversation_id [String] ID of conversation to remove member from + @param user_id [String] ID of member to be removed from the the conversation + @param callback [Function] Function to be called on server return + ### + delete_members: (conversation_id, user_id, callback) -> + @client.send_request + url: @client.create_url "/conversations/#{conversation_id}/members/#{user_id}" + type: 'DELETE' + callback: callback + + ### + Delete events from a conversation. + + @param message_id [String] ID of conversation to remove message from + @param primary_key [String] ID of the actual message + ### + delete_message_from_db: (conversation_id, message_id) -> + @storage_service.db[@storage_service.OBJECT_STORE_CONVERSATION_EVENTS] + .where 'raw.conversation' + .equals conversation_id + .and (record) -> record.mapped?.id is message_id + .delete() + + ### + Delete events from a conversation. + + @param conversation_id [String] delete message for this conversation + ### + delete_messages_from_db: (conversation_id) -> + @storage_service.db[@storage_service.OBJECT_STORE_CONVERSATION_EVENTS] + .where 'raw.conversation' + .equals conversation_id + .delete() + + ### + Delete events from a conversation. + + @param primary_key [String] Primary key used to find an event in the database + ### + update_asset_as_uploaded_in_db: (primary_key, asset_data) -> + @storage_service.load @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key + .then (record) => + record.mapped.data.id = asset_data.id + record.mapped.data.otr_key = asset_data.otr_key + record.mapped.data.sha256 = asset_data.sha256 + record.mapped.data.status = z.assets.AssetTransferState.UPLOADED + @storage_service.update @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key, record + .then => + @logger.log 'Updated asset message_et (uploaded)', primary_key + + ### + Delete events from a conversation. + + @param primary_key [String] Primary key used to find an event in the database + ### + update_asset_preview_in_db: (primary_key, asset_data) -> + @storage_service.load @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key + .then (record) => + record.mapped.data.preview_id = asset_data.id + record.mapped.data.preview_otr_key = asset_data.otr_key + record.mapped.data.preview_sha256 = asset_data.sha256 + @storage_service.update @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key, record + .then => + @logger.log 'Updated asset message_et (preview)', primary_key + + ### + Delete events from a conversation. + + @param primary_key [String] Primary key used to find an event in the database + ### + update_asset_as_failed_in_db: (primary_key, reason) -> + @storage_service.load @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key + .then (record) => + record.mapped.data.status = z.assets.AssetTransferState.UPLOAD_FAILED + record.mapped.data.reason = reason + @storage_service.update @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key, record + .then => + @logger.log 'Updated asset message_et (failed)', primary_key + + ### + Loads conversation states from the local database. + + ### + load_conversation_states_from_db: => + return new Promise (resolve, reject) => + @storage_service.get_all @storage_service.OBJECT_STORE_CONVERSATIONS + .then (conversation_states) => + @logger.log @logger.levels.INFO, "Loaded '#{conversation_states.length}' local conversation states", conversation_states + resolve conversation_states + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to load local conversation states', error + reject error + + ### + Load conversation events. + + @param conversation_id [String] ID of conversation + @param offset [String] Timestamp that loaded events have to undercut + @param limit [Number] Amount of events to load + @return [Promise] Promise that resolves with the retrieved records + ### + load_events_from_db: (conversation_id, offset, limit = z.config.MESSAGES_FETCH_LIMIT) -> + return new Promise (resolve, reject) => + @storage_service.db[@storage_service.OBJECT_STORE_CONVERSATION_EVENTS] + .where 'raw.conversation' + .equals conversation_id + .reverse() + .sortBy 'meta.timestamp' + .then (records) -> + records = (record for record in records when record.meta.timestamp < offset) if offset + has_further_events = records.length > limit + resolve [records.slice(0, limit), has_further_events] + .catch (error) => + @logger.log @logger.levels.ERROR, + "Failed to get events for conversation '#{conversation_et.id}': #{error.message}", error + reject error + + ### + Load all unread events of a conversation. + + @param conversation_et [z.entity.Conversation] Conversation entity + @param offset [String] Timestamp that loaded events have to undercut + @return [Promise] Promise that resolves with the retrieved records + ### + load_unread_events_from_db: (conversation_et, offset) -> + return new Promise (resolve, reject) => + conversation_id = conversation_et.id + @storage_service.db[@storage_service.OBJECT_STORE_CONVERSATION_EVENTS] + .where 'raw.conversation' + .equals conversation_id + .reverse() + .sortBy 'raw.time' + .then (records) -> + records = (record for record in records when record.meta.timestamp < offset) if offset + records = (record for record in records when record.meta.timestamp > conversation_et.last_read_timestamp()) + resolve records + .catch (error) => + @logger.log @logger.levels.ERROR, + "Failed to get unread events for conversation '#{conversation_et.id}': #{error.message}", error + reject error + + ### + Add users to an existing conversation. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/addMembers + + @param conversation_id [String] ID of conversation to add users to + @param user_ids [Array] IDs of users to be added to the conversation + @return [Promise] Promise that resolves with the server response + ### + post_members: (conversation_id, user_ids) -> + @client.send_json + url: @client.create_url "/conversations/#{conversation_id}/members" + type: 'POST' + data: + users: user_ids + + ### + Post an encrypted message to a conversation. + @note If "recipients" are not specified you will receive a list of all missing OTR recipients (user-client-map). + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/postOtrMessage + @example How to send "recipients" payload + "recipients": { + "": { + "": "" + } + } + + @param conversation_id [String] ID of conversation to send message in + @param payload [Object] Payload to be posted + @option [OtrRecipients] recipients Map with per-recipient data + @option [String] sender Client ID of the sender + @param force_sending [Boolean] Should the backend ignore missing clients + @return [Promise] Promise that resolve when the message was sent + ### + post_encrypted_message: (conversation_id, payload, force_sending) -> + url = @client.create_url "/conversations/#{conversation_id}/otr/messages" + url = "#{url}?ignore_missing=true" if force_sending + + @client.send_json + url: url + type: 'POST' + data: payload + + ### + Saves or updates a conversation entity in the local database. + + @param conversation_et [z.entity.Conversation] Conversation entity + @param updated_field [z.conversation.ConversationUpdateType] Property of the conversation entity which needs to be updated in the local database + @return [Promise] Promise which resolves with the conversation entity (if update was successful) or the conversation entity (if it was a new entity) + ### + save_conversation_in_db: (conversation_et, updated_field) => + if updated_field + @_update_conversation_in_db conversation_et, updated_field + else + @_save_conversation_in_db conversation_et + + ### + Update conversation properties. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/updateConversation + + @param conversation_id [String] ID of conversation to rename + @param name [String] New conversation name + @param callback [Function] Function to be called on server return + ### + update_conversation_properties: (conversation_id, name, callback) -> + @client.send_json + url: @client.create_url "/conversations/#{conversation_id}" + type: 'PUT' + data: + name: name + callback: callback + + ### + Update self membership properties. + + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/conversations/updateSelf + + @param conversation_id [String] ID of conversation to update + @param payload [Object] Updated properties + @param callback [Function] Function to be called on server return + ### + update_member_properties: (conversation_id, payload, callback) -> + @client.send_json + url: @client.create_url "/conversations/#{conversation_id}/self" + type: 'PUT' + data: payload + callback: (response, error) -> + if callback? + callback + conversation: conversation_id + data: data + , error diff --git a/app/script/conversation/ConversationStatus.coffee b/app/script/conversation/ConversationStatus.coffee new file mode 100644 index 00000000000..3563f85217e --- /dev/null +++ b/app/script/conversation/ConversationStatus.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Enum of a user's participation status in a conversation. +z.conversation.ConversationStatus = + CURRENT_MEMBER: 0 + PAST_MEMBER: 1 diff --git a/app/script/conversation/ConversationType.coffee b/app/script/conversation/ConversationType.coffee new file mode 100644 index 00000000000..8a844115a98 --- /dev/null +++ b/app/script/conversation/ConversationType.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Enum of different conversation types. +z.conversation.ConversationType = + REGULAR: 0 + SELF: 1 + ONE2ONE: 2 + CONNECT: 3 diff --git a/app/script/conversation/ConversationUnreadType.coffee b/app/script/conversation/ConversationUnreadType.coffee new file mode 100644 index 00000000000..72e4ae76cf5 --- /dev/null +++ b/app/script/conversation/ConversationUnreadType.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Enum of different types of unread indicators for a conversation. +z.conversation.ConversationUnreadType = + UNREAD: 'unread' + PING: 'ping' + CONNECT: 'connect' + MISSED_CALL: 'missed-call' diff --git a/app/script/conversation/ConversationUpdateType.coffee b/app/script/conversation/ConversationUpdateType.coffee new file mode 100644 index 00000000000..94739892ccd --- /dev/null +++ b/app/script/conversation/ConversationUpdateType.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +z.conversation.ConversationUpdateType = + ARCHIVED_STATE: 'archived_state' + ARCHIVED_TIMESTAMP: 'archived_timestamp' + CLEARED_TIMESTAMP: 'cleared_timestamp' + LAST_EVENT_TIMESTAMP: 'last_event_timestamp' + LAST_READ_TIMESTAMP: 'last_read_timestamp' + MUTED_STATE: 'mute_state' + MUTED_TIMESTAMP: 'muted_timestamp' diff --git a/app/script/conversation/EventMapper.coffee b/app/script/conversation/EventMapper.coffee new file mode 100644 index 00000000000..d55eca4b7e4 --- /dev/null +++ b/app/script/conversation/EventMapper.coffee @@ -0,0 +1,452 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.conversation ?= {} + +# Event Mapper to convert all server side JSON events into core entities. +class z.conversation.EventMapper + ### + Construct a new Event Mapper. + + @param asset_service [z.assets.AssetService] Asset handling service + ### + constructor: (@asset_service) -> + @logger = new z.util.Logger 'z.conversation.EventMapper', z.config.LOGGER.OPTIONS + + ### + Convert multiple JSON events into message entities. + + @param json [Object] Event data + @param conversation_et [z.entity.Conversation] Conversation entity the events belong to + + @return [Array] Mapped message entities + ### + map_json_events: (json, conversation_et) -> + events = (@map_json_event event, conversation_et for event in json.events.reverse() when event isnt undefined) + return events.filter (x) -> x isnt undefined + + map_json_event: (event, conversation_et) => + try + return @_map_json_event event, conversation_et + catch error + @logger.log @logger.levels.ERROR, 'Cannot map event', {error: error, event: event} + return undefined + + ### + Convert JSON event into a message entity. + + @param event [Object] Event data + @param conversation_et [z.entity.Conversation] Conversation entity the events belong to + + @return [z.entity.Message] Mapped message entity + ### + _map_json_event: (event, conversation_et) -> + switch event.type + when z.event.Backend.CONVERSATION.ASSET_META + message_et = @_map_event_asset_meta event + when z.event.Backend.CONVERSATION.ASSET_ADD + message_et = @_map_event_asset_add event + when z.event.Backend.CONVERSATION.MESSAGE_ADD + message_et = @_map_event_message_add event + when z.event.Backend.CONVERSATION.MEMBER_JOIN + message_et = @_map_event_member_join event, conversation_et + when z.event.Backend.CONVERSATION.MEMBER_LEAVE + message_et = @_map_event_member_leave event + when z.event.Backend.CONVERSATION.MEMBER_UPDATE + message_et = @_map_event_member_update() + when z.event.Backend.CONVERSATION.RENAME + message_et = @_map_event_rename event + when z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE + message_et = @_map_event_voice_channel_activate() + when z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE + message_et = @_map_event_voice_channel_deactivate event + when z.event.Backend.CONVERSATION.KNOCK + message_et = @_map_event_ping event + when z.event.Backend.CONVERSATION.LOCATION + message_et = @_map_event_location event + when z.event.Client.CONVERSATION.UNABLE_TO_DECRYPT + message_et = @_map_system_event_unable_to_decrypt event + else + message_et = @_map_event_ignored() + + message_et.id = event.id + message_et.type = event.type + message_et.from = event.from + message_et.timestamp = new Date(event.time).getTime() + message_et.primary_key = "#{conversation_et.id}@#{message_et.from}@#{message_et.timestamp}" + + if window.isNaN message_et.timestamp + @logger.log @logger.levels.ERROR, 'Could not get timestamp for message', event + + return message_et + + ############################################################################### + # Event mappers + ############################################################################### + + ### + Maps JSON data of conversation.asset_add message into message entity + + @private + + @param event [Object] Message data + + @return [z.entity.NormalMessage] Normal message entity + ### + _map_event_asset_add: (event) -> + message_et = new z.entity.ContentMessage() + if event.data?.info.tag is z.assets.ImageSizeType.PREVIEW + message_et.assets.push @_map_asset_preview_image event.data + if event.data?.info.tag is z.assets.ImageSizeType.MEDIUM + message_et.assets.push @_map_asset_medium_image event + message_et.nonce = event.data.info.nonce + return message_et + + ### + Maps JSON data of conversation.asset_add message into message entity + + @private + + @param event [Object] Message data + + @return [z.entity.NormalMessage] Normal message entity + ### + _map_event_asset_meta: (event) -> + message_et = new z.entity.ContentMessage() + message_et.assets.push @_map_asset_file event + message_et.nonce = event.data.info.nonce + return message_et + + ### + Maps JSON data of conversation.connect_request message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.NormalMessage] Normal message entity + ### + _map_event_connect_request: (event) -> + message_et = new z.entity.ContentMessage() + asset_et = @_map_asset_text event.data + message_et.visible false if not asset_et.text + message_et.assets.push asset_et + return message_et + + ### + Maps JSON data of other message types currently ignored into message entity + + @private + + @return [z.entity.SpecialMessage] Special message entity + ### + _map_event_ignored: -> + message_et = new z.entity.SystemMessage() + message_et.visible false + return message_et + + ### + Maps JSON data of conversation.member_join message into message entity + + @private + + @param event [Object] Message data received as JSON + @param conversation_et [z.entity.conversation] Conversation entity the event belongs to + + @return [z.entity.MemberMessage] Member message entity + ### + _map_event_member_join: (event, conversation_et) -> + message_et = new z.entity.MemberMessage() + if conversation_et.type() in [z.conversation.ConversationType.CONNECT, z.conversation.ConversationType.ONE2ONE] + if event.from is conversation_et.creator and event.data.user_ids.length is 1 + message_et.member_message_type = z.message.SystemMessageType.CONNECTION_ACCEPTED + event.data.user_ids = conversation_et.participating_user_ids() + else + message_et.visible false + else + creator_index = event.data.user_ids.indexOf event.from + if event.from is conversation_et.creator and creator_index isnt -1 + event.data.user_ids.splice creator_index, 1 + message_et.member_message_type = z.message.SystemMessageType.CONVERSATION_CREATE + + message_et.user_ids event.data.user_ids + + return message_et + + ### + Maps JSON data of conversation.member_join message into message entity + + @private + + @return [z.entity.MemberMessage] Member message entity + ### + _map_event_member_leave: (event) -> + message_et = new z.entity.MemberMessage() + message_et.user_ids event.data.user_ids + return message_et + + ### + Maps JSON data of conversation.member_update message into message entity + + @private + + @return [z.entity.MemberMessage] Member message entity + ### + _map_event_member_update: -> + message_et = new z.entity.MemberMessage() + return message_et + + ### + Maps JSON data of conversation.message_add message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.NormalMessage] Normal message entity + ### + _map_event_message_add: (event) -> + message_et = new z.entity.ContentMessage() + message_et.assets.push @_map_asset_text event.data + message_et.nonce = event.data.nonce + return message_et + + ### + Maps JSON data of conversation.knock message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.PingMessage] Ping message entity + ### + _map_event_ping: (event) -> + message_et = new z.entity.PingMessage() + message_et.nonce = event.data.nonce + return message_et + + ### + Maps JSON data of conversation.location message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.LocationMessage] Location message entity + ### + _map_event_location: (event) -> + message_et = new z.entity.ContentMessage() + asset_et = new z.entity.Location() + asset_et.longitude = event.data.location.longitude + asset_et.latitude = event.data.location.latitude + asset_et.name = event.data.location.name + asset_et.zoom = event.data.location.zoom + message_et.assets.push asset_et + message_et.nonce = event.data.nonce + return message_et + + ### + Maps JSON data of conversation.rename message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.RenameMessage] Rename message entity + ### + _map_event_rename: (event) -> + message_et = new z.entity.RenameMessage() + message_et.name = event.data.name + return message_et + + ### + Maps JSON data of conversation.voice-channel-activate message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.CallMessage] Call message entity + ### + _map_event_voice_channel_activate: -> + message_et = new z.entity.CallMessage() + message_et.call_message_type = z.message.CallMessageType.ACTIVATED + message_et.visible false + return message_et + ### + Maps JSON data of conversation.voice-channel-deactivate message into message entity + + @private + + @param event [Object] Message data received as JSON + + @return [z.entity.CallMessage] Call message entity + ### + _map_event_voice_channel_deactivate: (event) -> + message_et = new z.entity.CallMessage() + message_et.call_message_type = z.message.CallMessageType.DEACTIVATED + message_et.finished_reason = event.data.reason + message_et.visible message_et.finished_reason is z.calling.enum.CallFinishedReason.MISSED + return message_et + + ############################################################################### + # Asset mappers + ############################################################################### + + ### + Maps JSON data of text asset into asset entity + + @private + + @param data [Object] Asset data received as JSON + + @return [z.entity.Text] Text asset entity + ### + _map_asset_text: (data) -> + asset_et = new z.entity.Text data.id + asset_et.key = "text##{data.nonce}" + asset_et.text = data.content or data.message + asset_et.previews @_map_link_previews data.previews + return asset_et + + ### + Map link previews + + @private + + @param previews [Array] base64 encoded proto previews + + @return [Array] + ### + _map_link_previews: (link_previews = []) -> + return link_previews + .map (encoded_link_preview) -> z.proto.LinkPreview.decode64 encoded_link_preview + .map (link_preview) => @_map_link_preview link_preview + .filter (link_preview_et) -> link_preview_et? + + ### + Map link preview + + @private + + @param preview [z.proto.LinkPreview] + + @return [z.entity.LinkPreview] + ### + _map_link_preview: (link_preview = {}) -> + switch link_preview.preview + when 'article' + article = link_preview.article + + link_preview_et = new z.entity.LinkPreview() + link_preview_et.title = article.title + link_preview_et.summary = article.summary + link_preview_et.permanent_url = article.permanent_url + link_preview_et.original_url = link_preview.url + link_preview_et.url_offset = link_preview.url_offset + + if article.image?.uploaded? + {asset_token, asset_id, otr_key, sha256} = article.image.uploaded + otr_key = new Uint8Array otr_key.toArrayBuffer() + sha256 = new Uint8Array sha256.toArrayBuffer() + link_preview_et.image_resource z.assets.AssetRemoteData.v3 asset_id, otr_key, sha256, asset_token + + return link_preview_et + + ### + Maps JSON data of preview image asset into asset entity + + @private + + @param data [Object] Asset data received as JSON + + @return [z.entity.PreviewImage] Preview image asset entity + ### + _map_asset_preview_image: (data) -> + asset_et = new z.entity.PreviewImage data.id + asset_et.correlation_id = data.info.correlation_id + asset_et.content_type = data.content_type + asset_et.encoded_data = data.data + asset_et.width = data.info.original_width + asset_et.height = data.info.original_height + asset_et.original_width = data.info.original_width + asset_et.original_height = data.info.original_height + return asset_et + + ### + Maps JSON data of medium image asset into asset entity + + @private + + @param data [Object] Asset data received as JSON + + @return [z.entity.MediumImage] Medium image asset entity + ### + _map_asset_medium_image: (data) -> + asset_et = new z.entity.MediumImage data.data.id + asset_et.correlation_id = data.data.info.correlation_id + asset_et.width = data.data.info.width + asset_et.height = data.data.info.height + asset_et.original_width = data.data.info.original_width + asset_et.original_height = data.data.info.original_height + asset_et.ratio = asset_et.original_height / asset_et.original_width + asset_et.resource z.assets.AssetRemoteData.v2 data.conversation, asset_et.id, data.data.otr_key, data.data.sha256 + asset_et.dummy_url = z.util.dummy_image asset_et.original_width, asset_et.original_height + return asset_et + + ### + Maps JSON data of file asset into asset entity + + @private + + @param data [Object] Asset data received as JSON + + @return [z.entity.MediumImage] File asset entity + ### + _map_asset_file: (data) -> + asset_et = new z.entity.File data.data.id + asset_et.correlation_id = data.data.info.correlation_id + asset_et.conversation_id = data.conversation + + # original + asset_et.file_size = data.data.content_length + asset_et.file_type = data.data.content_type + asset_et.file_name = data.data.info.name + asset_et.meta = data.data.meta + asset_et.original_resource z.assets.AssetRemoteData.v2 asset_et.conversation_id, asset_et.id, data.data.otr_key, data.data.sha256, + if data.data.preview_id? + asset_et.preview_resource z.assets.AssetRemoteData.v2 asset_et.conversation_id, data.data.preview_id, data.data.preview_otr_key, data.data.preview_sha256 + asset_et.status data.data.status or z.assets.AssetTransferState.UPLOADING # TODO + return asset_et + + ### + Maps JSON data of local decrypt errors to message entity + + @private + + @param data [Object] Error data received as JSON + + @return [z.entity.MediumImage] Medium image asset entity + ### + _map_system_event_unable_to_decrypt: (event) -> + message_et = new z.entity.DecryptErrorMessage() + # error_code style "3690 (f0c0272e8f053774)" + message_et.error_code = event.error_code?.substring(0, 4) + message_et.client_id = event.error_code?.substring(5).replace(/[()]/g, '') + return message_et diff --git a/app/script/cryptography/CryptographyError.coffee b/app/script/cryptography/CryptographyError.coffee new file mode 100644 index 00000000000..ce4b43abed1 --- /dev/null +++ b/app/script/cryptography/CryptographyError.coffee @@ -0,0 +1,38 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cryptography ?= {} + +class z.cryptography.CryptographyError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @stack = (new Error()).stack + @type = type + + @:: = new Error() + @::constructor = @ + @::TYPE = + BROKEN_EXTERNAL: 'z.cryptography.CryptographyError::TYPE.BROKEN_EXTERNAL' + IGNORED_ASSET: 'z.cryptography.CryptographyError::TYPE.IGNORED_ASSET' + IGNORED_HOT_KNOCK: 'z.cryptography.CryptographyError::TYPE.IGNORED_HOT_KNOCK' + IGNORED_PREVIEW: 'z.cryptography.CryptographyError::TYPE.IGNORED_PREVIEW' + MISSING_MESSAGE: 'z.cryptography.CryptographyError::TYPE.MISSING_MESSAGE' + PREVIOUSLY_STORED: 'z.cryptography.CryptographyError::TYPE.PREVIOUSLY_STORED' + UNHANDLED_TYPE: 'z.cryptography.CryptographyError::TYPE.UNHANDLED_TYPE' diff --git a/app/script/cryptography/CryptographyErrorType.coffee b/app/script/cryptography/CryptographyErrorType.coffee new file mode 100644 index 00000000000..cf05dc887f3 --- /dev/null +++ b/app/script/cryptography/CryptographyErrorType.coffee @@ -0,0 +1,30 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cryptography ?= {} + +z.cryptography.CryptographyErrorType = + DUPLICATE_MESSAGE: '1701' + INVALID_MESSAGE_SESSION_NOT_MATCHING: '1976' + INVALID_MESSAGE_SESSION_MISSING: '2237' + INVALID_SIGNATURE: '8550' + OUTDATED_MESSAGE: '2521' + PRE_KEY_NOT_FOUND: '3337' + REMOTE_IDENTITY_CHANGED: '3690' + TOO_DISTANT_FUTURE: '1300' diff --git a/app/script/cryptography/CryptographyMapper.coffee b/app/script/cryptography/CryptographyMapper.coffee new file mode 100644 index 00000000000..be11a719f15 --- /dev/null +++ b/app/script/cryptography/CryptographyMapper.coffee @@ -0,0 +1,228 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cryptography ?= {} + +# Cryptography Mapper to convert all server side JSON encrypted events into core entities +class z.cryptography.CryptographyMapper + # Construct a new Cryptography Mapper. + constructor: -> + @logger = new z.util.Logger 'z.cryptography.CryptographyMapper', z.config.LOGGER.OPTIONS + + ### + OTR to JSON mapper. + @param generic_message [z.proto.GenericMessage] Received ProtoBuffer message + @param event [z.event.Backend.CONVERSATION.OTR-ASSET-ADD, z.event.Backend.CONVERSATION.OTR-MESSAGE-ADD] Event + @return [Object] Promise that resolves with the mapped event + ### + map_generic_message: (generic_message, event) => + mapped = undefined + + Promise.resolve() + .then -> + if not generic_message + error_message = "Failed to map OTR event '#{event.id}' as decrypted generic message is missing" + throw new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.MISSING_MESSAGE + .then -> + mapped = + conversation: event.conversation + id: generic_message.message_id + from: event.from + time: event.time + .then => + return @_map_generic_message generic_message, event + .then (specific_content) -> + $.extend mapped, specific_content + return mapped + + _map_generic_message: (generic_message, event) -> + switch generic_message.content + when 'asset' + return @_map_asset generic_message.asset, generic_message.message_id, event.data?.id + when 'cleared' + return @_map_cleared generic_message.cleared + when 'deleted' + return @_map_deleted generic_message.deleted + when 'external' + return @_map_external generic_message.external, event + when 'image' + return @_map_image generic_message.image, event.data.id + when 'knock' + return @_map_knock generic_message.knock, generic_message.message_id + when 'lastRead' + return @_map_last_read generic_message.lastRead + when 'location' + return @_map_location generic_message.location, generic_message.message_id + when 'text' + return @_map_text generic_message.text, generic_message.message_id + else + error_message = "Skipped event '#{generic_message.message_id}' of unhandled type '#{generic_message.content}'" + throw new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.UNHANDLED_TYPE + + _map_asset: (asset, event_nonce, event_id) -> + if asset.uploaded? + return @_map_asset_uploaded asset.uploaded, event_id + else if asset.not_uploaded? + return @_map_asset_not_uploaded asset.not_uploaded + else if asset.preview? + return @_map_asset_preview asset.preview, event_id + else if asset.original? + return @_map_asset_original asset.original, event_nonce + else + error_message = 'Ignored asset preview' + @logger.log @logger.levels.INFO, error_message + throw new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.IGNORED_ASSET + + _map_asset_meta_data: (original) -> + if original.audio? + return {} = + duration: original.audio.duration_in_millis.toNumber() / 1000 + loudness: new Uint8Array(original.audio.normalized_loudness?.toArrayBuffer() or []) + + _map_asset_original: (original, event_nonce) -> + return {} = + data: + content_length: original.size.toNumber() + content_type: original.mime_type + info: + name: original.name + nonce: event_nonce + meta: @_map_asset_meta_data(original) + type: z.event.Backend.CONVERSATION.ASSET_META + + _map_asset_not_uploaded: (not_uploaded) -> + return {} = + data: + reason: not_uploaded + type: z.event.Backend.CONVERSATION.ASSET_UPLOAD_FAILED + + _map_asset_uploaded: (uploaded, event_id) -> + return {} = + data: + id: event_id + otr_key: new Uint8Array uploaded.otr_key?.toArrayBuffer() + sha256: new Uint8Array uploaded.sha256?.toArrayBuffer() + type: z.event.Backend.CONVERSATION.ASSET_UPLOAD_COMPLETE + + _map_asset_preview: (preview, event_id) -> + return {} = + data: + id: event_id + otr_key: new Uint8Array preview.remote.otr_key?.toArrayBuffer() + sha256: new Uint8Array preview.remote.sha256?.toArrayBuffer() + type: z.event.Backend.CONVERSATION.ASSET_PREVIEW + + _map_cleared: (cleared) -> + return {} = + conversation: cleared.conversation_id + data: + cleared_timestamp: cleared.cleared_timestamp.toString() + type: z.event.Backend.CONVERSATION.MEMBER_UPDATE + + _map_deleted: (deleted) -> + return {} = + data: + conversation_id: deleted.conversation_id + message_id: deleted.message_id + type: z.event.Backend.CONVERSATION.MESSAGE_DELETE + + ### + Unpacks a specific generic message which is wrapped inside an external generic message. + + @note Wrapped messages get the 'message_id' of their wrappers (external message) + @param external [z.proto.GenericMessage] Generic message of type 'external' + @param event [JSON] Backend event of type 'conversation.otr-message-add' + ### + _map_external: (external, event) -> + data = + text: z.util.base64_to_array event.data.data + otr_key: new Uint8Array external.otr_key.toArrayBuffer() + sha256: new Uint8Array external.sha256.toArrayBuffer() + + z.assets.AssetCrypto.decrypt_aes_asset data.text.buffer, data.otr_key.buffer, data.sha256.buffer + .then (external_message_buffer) -> + return z.proto.GenericMessage.decode external_message_buffer + .then (generic_message) => + @logger.log @logger.levels.INFO, "Received external message of type '#{generic_message.content}'", generic_message + return @_map_generic_message generic_message, event + .catch (error) -> + throw new z.cryptography.CryptographyError error.message, z.cryptography.CryptographyError::TYPE.BROKEN_EXTERNAL + + _map_image: (image, event_id) -> + if image.tag is 'medium' + return @_map_image_medium image, event_id + else + error_message = 'Ignored image preview' + @logger.log @logger.levels.INFO, error_message + throw new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.IGNORED_PREVIEW + + _map_image_medium: (image, event_id) -> + return {} = + data: + content_length: image.size + content_type: image.mime_type + id: event_id + info: + tag: image.tag + width: image.width + height: image.height + nonce: event_id + original_width: image.original_width + original_height: image.original_height + public: false + otr_key: new Uint8Array image.otr_key?.toArrayBuffer() + sha256: new Uint8Array image.sha256?.toArrayBuffer() + type: z.event.Backend.CONVERSATION.ASSET_ADD + + _map_knock: (knock, event_id) -> + if knock.hot_knock + error_message = 'Ignored hot knock' + @logger.log @logger.levels.INFO, error_message + throw new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.IGNORED_HOT_KNOCK + else + return {} = + data: + nonce: event_id + type: z.event.Backend.CONVERSATION.KNOCK + + _map_last_read: (last_read) -> + return {} = + conversation: last_read.conversation_id + data: + last_read_timestamp: last_read.last_read_timestamp.toString() + type: z.event.Backend.CONVERSATION.MEMBER_UPDATE + + _map_location: (location, event_id) -> + return {} = + data: + location: + longitude: location.longitude + latitude: location.latitude + name: location.name + zoom: location.zoom + nonce: event_id + type: z.event.Backend.CONVERSATION.LOCATION + + _map_text: (text, event_id) -> + return {} = + data: + content: "#{text.content}" + nonce: event_id + previews: text.link_preview.map (preview) -> preview.encode64() + type: z.event.Backend.CONVERSATION.MESSAGE_ADD diff --git a/app/script/cryptography/CryptographyRepository.coffee b/app/script/cryptography/CryptographyRepository.coffee new file mode 100644 index 00000000000..7b3c8e9e6b6 --- /dev/null +++ b/app/script/cryptography/CryptographyRepository.coffee @@ -0,0 +1,625 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cryptography ?= {} + +# Cryptography repository for all cryptography interactions with the cryptography service. +class z.cryptography.CryptographyRepository + ### + Construct a new Cryptography repository. + @param cryptography_service [z.cryptography.CryptographyService] Backend REST API cryptography service implementation + @param storage_repository [z.storage.StorageRepository] Repository for all storage interactions + ### + constructor: (@cryptography_service, @storage_repository) -> + @logger = new z.util.Logger 'z.cryptography.CryptographyRepository', z.config.LOGGER.OPTIONS + + @cryptography_mapper = new z.cryptography.CryptographyMapper() + + @current_client = undefined + @cryptobox = undefined + return @ + + + ############################################################################### + # Initialization + ############################################################################### + + ### + Initialize the repository. + @return [Promise] Promise that will resolve with the repository after initialization + ### + init: => + return Promise.resolve() + .then => + @logger.log @logger.levels.INFO, "Initialize Cryptobox with our storage repository on '#{@storage_repository.storage_service.db_name}'", @storage_repository + @cryptobox = new cryptobox.Cryptobox @storage_repository + @logger.log @logger.levels.INFO, 'Initialized repository' + return @ + + + ############################################################################### + # Pre-keys + ############################################################################### + + ### + Generate all keys need for client registration. + @return [Promise] Promise that resolves with an array of last resort key, pre-keys, and signaling keys + ### + generate_client_keys: => + return new Promise (resolve, reject) => + last_resort_key = undefined + pre_keys = undefined + signaling_keys = undefined + + @_generate_last_resort_key() + .then (key) => + last_resort_key = key + @logger.log @logger.levels.INFO, 'Generated last resort key', last_resort_key + return @_generate_pre_keys() + .then (keys) => + pre_keys = keys + @logger.log @logger.levels.INFO, "Number of generated pre-keys: #{pre_keys.length}", pre_keys + return @_generate_signaling_keys() + .then (keys) => + signaling_keys = keys + @logger.log @logger.levels.INFO, 'Generated signaling keys', signaling_keys + resolve [last_resort_key, pre_keys, signaling_keys] + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to generate client keys: #{error.message}", error + reject error + + ### + Get a pre-key for a user client. + + @param user_id [String] ID of user + @param client_id [String] ID of client to request pre-key for + @return [Promise] Promise that resolves with a pre-key for the client + ### + get_user_pre_key: (user_id, client_id) -> + @cryptography_service.get_user_pre_key user_id, client_id + .then (response) -> + return response.prekey + .catch (error) -> + if error.code is z.service.BackendClientError::STATUS_CODE.NOT_FOUND + throw new z.user.UserError 'Pre-key not found', z.user.UserError::TYPE.PRE_KEY_NOT_FOUND + else + error_message = "Failed to get pre-key from backend: #{error.message}" + throw new z.user.UserError error_message, z.user.UserError::TYPE.REQUEST_FAILURE + + ### + Get a pre-key for client of in the user client map. + @param user_client_map [Object] User client map to request pre-keys for + @return [Promise] Promise that resolves a map of pre-keys for the requested clients + ### + get_users_pre_keys: (user_client_map) -> + @cryptography_service.get_users_pre_keys user_client_map + .then (response) -> + return response + .catch (error) -> + error_message = "Failed to get pre-key from backend: #{error.message}" + throw new z.user.UserError error_message, z.user.UserError::TYPE.REQUEST_FAILURE + + ### + Construct the pre-key. + + @private + @param id [String] ID of pre-key to be constructed + @return [Promise} Promise that will resolve with the new pre-key as object + ### + _construct_pre_key_promise: (id) => + @cryptobox.new_prekey id + .then (pre_key_bundle) -> + pre_key_model = + id: id + key: z.util.array_to_base64 pre_key_bundle + return pre_key_model + + ### + Construct the last resort pre-key. + @private + @return [Promise} Promise that will resolve with the new pre-key as object + ### + _generate_last_resort_key: => + return @_construct_pre_key_promise Proteus.keys.PreKey.MAX_PREKEY_ID + + ### + Generate the pre-keys. + + @private + @return [Promise} Promise that will resolve with an arrays of all the generated new pre-keys as object + ### + _generate_pre_keys: => + return Promise.all (@_construct_pre_key_promise i for i in [0...1]) + + ### + Generate the signaling keys + + @private + @return [Object] Object containing the signaling keys + ### + _generate_signaling_keys: -> + random_bytes = new Uint8Array sodium.crypto_auth_hmacsha256_KEYBYTES + crypto.getRandomValues random_bytes + + hmac = sodium.crypto_auth_hmacsha256 random_bytes, sodium.crypto_hash_sha256 'salt' + encryption_key = sodium.to_base64 hmac + mac_key = sodium.to_base64 hmac + + signaling_keys = + enckey: encryption_key + mackey: mac_key + + return signaling_keys + + + ############################################################################### + # Sessions + ############################################################################### + + ### + Get session. + @param user_id [String] User ID + @param client_id [String] ID of client to retrieve session for + @return [Promise] Promise that resolves with the session + ### + get_session: (user_id, client_id) -> + return Promise.resolve() + .then => + return @load_session(user_id, client_id) or @_initiate_new_session user_id, client_id + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to get session for client '#{client_id}' of user '#{user_id}': #{error.message}", error + + ### + Get sessions. + @param user_client_map [Object] User client map to get sessions for + @return [Promise>] Promise that resolves with an array of sessions + ### + get_sessions: (user_client_map, use_local_sessions = true) => + return new Promise (resolve, reject) => + cryptobox_session_map = {} + missing_session_map = {} + + if use_local_sessions + [cryptobox_session_map, missing_session_map] = @_get_sessions_local user_client_map + @logger.log @logger.levels.INFO, "Found local sessions for '#{Object.keys(cryptobox_session_map).length}' users", cryptobox_session_map + else + missing_session_map = user_client_map + + @_get_sessions_missing cryptobox_session_map, missing_session_map + .then (cryptobox_session_map) -> + resolve cryptobox_session_map + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to get sessions: #{error.message}", user_client_map + reject error + + ### + Loads the session from Cryptobox. + + @param user_id [String] ID of user + @param client_id [String] ID of client to retrieve session for + @return [cryptobox.CryptoboxSession] Retrieved session + ### + load_session: (user_id, client_id) => + session_id = @_construct_session_id user_id, client_id + return @cryptobox.session_load session_id + + ### + Resets a session. + + @param user_id [String] User ID of our chat partner + @param client_id [String] Client ID of our chat partner + @return [Promise] Promise that will resolve with the ID of the reset session + ### + reset_session: (user_id, client_id) => + return Promise.resolve() + .then => + cryptobox_session = @load_session user_id, client_id + + if cryptobox_session + @logger.log @logger.levels.INFO, "Deleting session for client '#{client_id}' of user '#{user_id}'", cryptobox_session + @cryptobox.session_delete cryptobox_session.id + .then -> return cryptobox_session.id + else + @logger.log @logger.levels.INFO, "We cannot delete the session for client '#{client_id}' of user '#{user_id}' because it was not found" + return undefined + + ### + Save a session. + @note Sessions MUST be saved AFTER encrypting messages, but BEFORE sending the encrypted message across the network. + @param session [cryptobox.CryptoboxSession] Session to be saved + ### + save_session: (cryptobox_session) => + @logger.log @logger.levels.INFO, "Persisting session '#{cryptobox_session.id}'", cryptobox_session + return @cryptobox.session_save cryptobox_session + + ### + Save sessions. + @param session [Array] Array of sessions to be saved + ### + save_sessions: (cryptobox_sessions) => + cryptobox_sessions.forEach (cryptobox_session) => @save_session cryptobox_session + + ### + Construct a session ID. + + @private + @param user_id [String] User ID for the remote participant + @param client_id [String] Client ID of the remote participant + @return [String] Client ID + ### + _construct_session_id: (user_id, client_id) -> + return "#{user_id}@#{client_id}" + + ### + Get local session for a user client map. + + @private + @param user_client_map [Object] User client map + @return [Array] An array containing two user client maps. The first contains the found sessions. + ### + _get_sessions_local: (user_client_map) -> + cryptobox_session_map = {} + missing_session_map = {} + for user_id, client_ids of user_client_map + cryptobox_session_map[user_id] = {} + for client_id in client_ids + session = @load_session user_id, client_id + + if session + cryptobox_session_map[user_id][client_id] = session + else + missing_session_map[user_id] ?= [] + missing_session_map[user_id].push client_id + return [cryptobox_session_map, missing_session_map] + + ### + Get missing session for a user client map. + + @private + @param user_client_map [Object] User client map + @param missing_session_map [Object] Client session map + @return [Promise] Promise that resolves with a client session map + ### + _get_sessions_missing: (cryptobox_session_map, missing_session_map) -> + if Object.keys(missing_session_map).length > 0 + @logger.log @logger.levels.INFO, "Missing sessions for '#{Object.keys(missing_session_map).length}' users", missing_session_map + return @_initiate_new_sessions cryptobox_session_map, missing_session_map + else + return Promise.resolve cryptobox_session_map + + ### + Initiate a new session for a given client. + + @private + @param user_id [String] User ID for the remote participant + @param client_id [String] Client ID of the remote participant + @return [Promise] Promise that resolves with the new session + ### + _initiate_new_session: (user_id, client_id) -> + @get_user_pre_key user_id, client_id + .then (pre_key) => + return @_session_from_prekey user_id, client_id, pre_key.key + .catch (error) => + if error instanceof z.user.UserError + if error.type is z.user.UserError::TYPE.PRE_KEY_NOT_FOUND + @logger.log @logger.levels.WARN, + "Client '#{client_id}' does no longer exist on the backend, should ignored and removed locally" + amplify.publish z.event.WebApp.CLIENT.DELETE, user_id, client_id + else if error.type is z.user.UserError::TYPE.REQUEST_FAILURE + @logger.log @logger.levels.WARN, + "Failed to request pre-key for client '#{client_id}' of user '#{user_id}'': #{error.message}", error + else + @logger.log @logger.levels.ERROR, + "Failed to initialize session from pre-key for client '#{client_id}' of user '#{user_id}': #{error.message}", error + return undefined + + ### + Initiate new sessions for a given map. + + @private + @param cryptobox_session_map [Object] User client map of containing the known sessions + @param user_client_map [Object] User client map of missing sessions + @return [Promise] Promise that resolves with a user client map containing the new sessions + ### + _initiate_new_sessions: (cryptobox_session_map, user_client_map) -> + @get_users_pre_keys user_client_map + .then (user_pre_key_map) => + @logger.log @logger.levels.INFO, "Fetched pre-keys for '#{Object.keys(user_pre_key_map).length}' users", user_pre_key_map + for user_id, client_pre_keys of user_pre_key_map + cryptobox_session_map[user_id] ?= {} + for client_id, pre_key of client_pre_keys when pre_key + cryptobox_session_map[user_id][client_id] = @_session_from_prekey user_id, client_id, pre_key.key + return cryptobox_session_map + .catch (error) => + if error.type is z.user.UserError::TYPE.REQUEST_FAILURE + @logger.log @logger.levels.WARN, + "Failed to request pre-keys for user '#{user_id}'': #{error.message}", error + throw error + + ### + Create a session from a message. + @private + @param user_id [String] User ID + @param client_id [String] ID of client to initialize session for + @param message [ArrayBuffer] Serialised OTR message + @return [cryptobox.CryptoboxSession] New cryptography session + ### + _session_from_message: (user_id, client_id, message) => + session_id = @_construct_session_id user_id, client_id + cryptobox_session = @cryptobox.session_from_message session_id, message + return cryptobox_session + + ### + Create a session from a pre-key. + @private + @param user_id [String] User ID + @param client_id [String] ID of client to initialize session for + @param serialized_pre_key_bundle [String] Base 64-encoded and serialized pre-key bundle + @return [cryptobox.CryptoboxSession] New cryptography session + ### + _session_from_prekey: (user_id, client_id, encoded_pre_key_bundle) => + decoded_pre_key_bundle = sodium.from_base64 encoded_pre_key_bundle + session_id = @_construct_session_id user_id, client_id + cryptobox_session = @cryptobox.session_from_prekey session_id, decoded_pre_key_bundle.buffer + return cryptobox_session + + + ############################################################################### + # Encryption + ############################################################################### + + ### + Bundles and encrypts the generic message for all given clients. + + @param user_client_map [Object] Contains all users and their known clients + @param generic_message [z.proto.GenericMessage] Proto buffer message to be encrypted + @return [Promise] Promise that resolves with the encrypted payload + ### + encrypt_generic_message: (user_client_map, generic_message, payload) => + return Promise.resolve() + .then => + if payload + use_local_sessions = false + @logger.log @logger.levels.INFO, 'Skip local sessions when encrypting message' + else + payload = @_construct_payload @current_client().id + use_local_sessions = true + @logger.log @logger.levels.INFO, 'Encrypt message using local sessions' + return @get_sessions user_client_map, use_local_sessions + .then (cryptobox_session_map) => + return @_add_payload_recipients payload, generic_message, cryptobox_session_map + + ### + Add the encrypted message for recipients to the payload message. + + @private + @param payload [Object] Payload to add encrypted message for recipients to + @param cryptobox_sessions [Array] Sessions for all the recipients of message + @param generic_message [z.proto.GenericMessage] ProtoBuffer message to be send + @return [Object] Payload to send to backend + ### + _add_payload_recipients: (payload, generic_message, cryptobox_session_map) -> + for user_id, client_session_map of cryptobox_session_map + payload.recipients[user_id] ?= {} + for client_id, cryptobox_session of client_session_map + payload.recipients[user_id][client_id] = @_encrypt_payload_for_session cryptobox_session, generic_message + return payload + + ### + Construct the payload for an encrypted message. + + @private + @param sender [String] Client ID of message sender + @return [Object] Payload to send to backend + ### + _construct_payload: (sender) -> + return {} = + sender: sender + recipients: {} + + ### + Encrypt the generic message for a given session. + @note We created the convention that whenever we fail to encrypt for a specific client, we send a bomb emoji (no fun!) + + @private + @param cryptobox_session [cryptobox.CryptoboxSession] Cryptographic session + @param generic_message [z.proto.GenericMessage] ProtoBuffer message + @return [String] Encrypted message as BASE64 encoded string + ### + _encrypt_payload_for_session: (cryptobox_session, generic_message) -> + try + generic_message_encrypted = cryptobox_session.encrypt generic_message.toArrayBuffer() + generic_message_encrypted_base64 = z.util.array_to_base64 generic_message_encrypted + @save_session cryptobox_session + return generic_message_encrypted_base64 + catch error + ids = z.client.Client.dismantle_user_client_id cryptobox_session.id + # Note: We created the convention that whenever we fail to encrypt for a specific client, we send a bomb emoji (no fun!) + @logger.log @logger.levels.ERROR, + "Could not encrypt OTR message of type '#{generic_message.content}' for user ID '#{ids.user_id}' with client ID '#{ids.client_id}': #{error.message}", error + return '💣' + + + ############################################################################### + # Decryption + ############################################################################### + + ### + @return [cryptobox.CryptoboxSession, z.proto.GenericMessage] Cryptobox session along with the decrypted message in ProtocolBuffer format + ### + decrypt_event: (event) => + return new Promise (resolve, reject) => + if not event.data + error_message = new Error "Encrypted event with ID '#{event.id}' does not contain its data payload" + @logger.log @logger.levels.ERROR, error_message, event + reject new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.MISSING_MESSAGE + return + + if event.type is z.event.Backend.CONVERSATION.OTR_ASSET_ADD + ciphertext = event.data.key + else if event.type is z.event.Backend.CONVERSATION.OTR_MESSAGE_ADD + ciphertext = event.data.text + + primary_key = @storage_repository.construct_primary_key event.conversation, event.from, event.time + @storage_repository.load_event_for_conversation primary_key + .then (loaded_event) => + if loaded_event is undefined + resolve @_decrypt_message event, ciphertext + else + error_message = "Skipped decryption of event '#{event.type}' (#{primary_key}) because it was previously stored" + @logger.log @logger.levels.INFO, error_message + reject new z.cryptography.CryptographyError error_message, z.cryptography.CryptographyError::TYPE.PREVIOUSLY_STORED + .catch (decrypt_error) => + # Get error information + receiving_client_id = event.data.recipient + remote_client_id = event.data.sender + remote_user_id = event.from + + # Handle error + if decrypt_error instanceof Proteus.errors.DecryptError.DuplicateMessage or decrypt_error instanceof Proteus.errors.DecryptError.OutdatedMessage + # We don't need to show duplicate message errors to the user + return resolve [undefined, undefined] + + else if decrypt_error instanceof Proteus.errors.DecryptError.InvalidMessage or decrypt_error instanceof Proteus.errors.DecryptError.InvalidSignature + # Session is broken, let's see what's really causing it... + session_id = @_construct_session_id remote_user_id, remote_client_id + @logger.log @logger.levels.ERROR, + "Session '#{session_id}' broken or out of sync. Reset the session and decryption is likely to work again.\r\n" + + "Try: wire.app.repository.cryptography.reset_session('#{remote_user_id}', '#{remote_client_id}');" + + if decrypt_error instanceof Proteus.errors.DecryptError.InvalidMessage + @logger.log @logger.levels.ERROR, + "Message is for client ID '#{receiving_client_id}' and we have client ID '#{@current_client().id}'." + + else if decrypt_error instanceof Proteus.errors.DecryptError.RemoteIdentityChanged + # Remote identity changed... Is there a man in the middle or do we mess up with clients? + session = @load_session remote_user_id, remote_client_id + remote_fingerprint = session.session.remote_identity.public_key.fingerprint() + + message = "Fingerprints do not match: We expect this fingerprint '#{remote_fingerprint}' from user ID '#{remote_user_id}' with client ID '#{remote_client_id}'" + @logger.log @logger.levels.ERROR, message, session + + # Show error in JS console + @logger.log @logger.levels.ERROR, + "Decryption of '#{event.type}' (#{primary_key}) failed: #{decrypt_error.message}", {error: decrypt_error, event: event} + + # Report error to Localytics and Raygun + hashed_error_message = z.util.murmurhash3 decrypt_error.message, 42 + error_code = hashed_error_message.toString().substr 0, 4 + @_report_decrypt_error event, decrypt_error, error_code + + unable_to_decrypt_event = + conversation: event.conversation + id: z.util.create_random_uuid() + type: z.event.Client.CONVERSATION.UNABLE_TO_DECRYPT + from: remote_user_id + time: event.time + error: "#{decrypt_error.message} (#{remote_client_id})" + error_code: "#{error_code} (#{remote_client_id})" + + # Show error message in message view + amplify.publish z.event.WebApp.EVENT.INJECT, unable_to_decrypt_event + + resolve [undefined, undefined] + + ### + Save an encrypted event. + + @note IMPORTANT: + Session should only be saved after the plaintext is saved somewhere safe, as, after saving, + the session will be unable to decrypt the message again. + + @param generic_message [z.proto.GenericMessage] Received ProtoBuffer message + @param event [JSON] JSON of 'z.event.Backend.CONVERSATION.OTR-ASSET-ADD' or 'z.event.Backend.CONVERSATION.OTR-MESSAGE-ADD' event + @return [Promise] Promise that will resolve with the saved record + ### + save_encrypted_event: (generic_message, event) => + @cryptography_mapper.map_generic_message generic_message, event + .then (mapped) => + primary_key = @storage_repository.construct_primary_key event.conversation, event.from, event.time + return @storage_repository.save_decrypted_conversation_event primary_key, event, mapped + .then (primary_key) => + @logger.log @logger.levels.INFO, "Saved '#{generic_message.content}' (OTR) with primary key '#{primary_key}'" + return @storage_repository.load_event_for_conversation primary_key + .catch (error) => + @logger.log @logger.levels.ERROR, "Saving encrypted message failed: #{error.message}", error + if error instanceof z.cryptography.CryptographyError + return undefined + else + throw error + + ### + Save an unencrypted event. + @param event [Object] JSON of unencrypted backend event + @return [Promise] Promise that will resolve with the saved record + ### + save_unencrypted_event: (event) -> + @storage_repository.save_unencrypted_conversation_event event + .then (primary_key) => + @logger.log @logger.levels.INFO, "Saved unencrypted event '#{event.type}' with primary key '#{primary_key}'" + @storage_repository.load_event_for_conversation primary_key + .catch (error) => + @logger.log @logger.levels.ERROR, "Saving unencrypted message failed: #{error.message}", error + throw error + + ### + @return [z.proto.GenericMessage] Decrypted message in ProtocolBuffer format + ### + _decrypt_message: (event, ciphertext) => + user_id = event.from + client_id = event.data.sender + + session = @load_session user_id, client_id + msg_bytes = sodium.from_base64(ciphertext).buffer + + decrypted_message = undefined + + if session + decrypted_message = session.decrypt msg_bytes + else + [session, decrypted_message] = @_session_from_message user_id, client_id, msg_bytes + + generic_message = z.proto.GenericMessage.decode decrypted_message + @save_session session + return generic_message + + ### + Report decryption error to Localytics and stack traces to Raygun. + @note We currently do not want to report duplicate message errors. + ### + _report_decrypt_error: (event, decrypt_error, error_code) => + remote_client_id = event.data.sender + remote_user_id = event.from + session_id = @_construct_session_id remote_user_id, remote_client_id + + attributes = + cause: "#{error_code}: #{decrypt_error.message}" + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.E2EE.CANNOT_DECRYPT_MESSAGE, attributes + + if decrypt_error not instanceof Proteus.errors.DecryptError.DuplicateMessage and decrypt_error not instanceof Proteus.errors.DecryptError.TooDistantFuture + custom_data = + client_local_class: @current_client().class + client_local_type: @current_client().type + error_code: error_code + event_type: event.type + session_id: session_id + + raygun_error = new Error "Decryption failed: #{decrypt_error.message}" + raygun_error.stack = decrypt_error.stack + Raygun.send raygun_error, custom_data diff --git a/app/script/cryptography/CryptographyService.coffee b/app/script/cryptography/CryptographyService.coffee new file mode 100644 index 00000000000..b7a59617170 --- /dev/null +++ b/app/script/cryptography/CryptographyService.coffee @@ -0,0 +1,55 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.cryptography ?= {} + +# Cryptography service for all cryptography related calls to the backend REST API. +class z.cryptography.CryptographyService + ### + Construct a new Cryptography Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.cryptography.CryptographyService', z.config.LOGGER.OPTIONS + + ### + Gets a pre-key for a client of a user + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/getPrekey + + @param user_id [String] User ID + @param client_id [String] Client ID + @return [Promise] Promise that resolves with a pre-key for the given client of the a user + ### + get_user_pre_key: (user_id, client_id) -> + @client.send_request + type: 'GET' + url: @client.create_url "/users/#{user_id}/prekeys/#{client_id}" + + ### + Gets a pre-key for each client of a user client map + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/getMultiPrekeyBundles + + @param user_client_map [Object] User client map to request pre-keys for + @return [Promise] Promise that resolves with a pre-key for each client of the given map + ### + get_users_pre_keys: (user_client_map) -> + @client.send_json + type: 'POST' + url: @client.create_url '/users/prekeys' + data: user_client_map diff --git a/app/script/entity/Connection.coffee b/app/script/entity/Connection.coffee new file mode 100644 index 00000000000..33a79c679bf --- /dev/null +++ b/app/script/entity/Connection.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +class z.entity.Connection + constructor: -> + @conversation_id = null + @from = null + @last_update = null + @message = null + @status = ko.observable z.user.ConnectionStatus.UNKNOWN + @to = null diff --git a/app/script/entity/Conversation.coffee b/app/script/entity/Conversation.coffee new file mode 100644 index 00000000000..8c577a34a2b --- /dev/null +++ b/app/script/entity/Conversation.coffee @@ -0,0 +1,511 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Conversation entity. +class z.entity.Conversation + ### + Construct a new conversation entity. + @param conversation_id [String] Conversation ID + ### + constructor: (conversation_id = '') -> + @id = conversation_id + @creator = undefined + @type = ko.observable() + @name = ko.observable() + @input = ko.observable z.storage.get_value("#{z.storage.StorageKey.CONVERSATION.INPUT}|#{@id}") or '' + @input.subscribe (text) => + z.storage.set_value "#{z.storage.StorageKey.CONVERSATION.INPUT}|#{@id}", text + + @is_pending = ko.observable false + @is_loaded = ko.observable false + + @all_user_ids = ko.observableArray [] + @participating_user_ets = ko.observableArray [] # Does not include us + @participating_user_ids = ko.observableArray [] + @self = undefined + @number_of_participants = ko.computed => return @participating_user_ids().length + + @is_group = ko.computed => @type() is z.conversation.ConversationType.REGULAR + @is_one2one = ko.computed => @type() is z.conversation.ConversationType.ONE2ONE + @is_request = ko.computed => @type() is z.conversation.ConversationType.CONNECT + @is_self = ko.computed => @type() is z.conversation.ConversationType.SELF + + # in case this is a one2one conversation this is the connection to that user + @connection = ko.observable new z.entity.Connection() + @connection.subscribe (connection_et) => @participating_user_ids [connection_et.to] + + ############################################################################### + # E2EE conversation states + ############################################################################### + + @archived_state = ko.observable false + @muted_state = ko.observable false + + @archived_timestamp = ko.observable 0 + @cleared_timestamp = ko.observable 0 + @last_event_timestamp = ko.observable 0 + @last_read_timestamp = ko.observable 0 + @muted_timestamp = ko.observable 0 + + ############################################################################### + # Conversation states for view + ############################################################################### + + @is_muted = ko.computed => + return @muted_state() + + @is_archived = ko.computed => + archived = @last_event_timestamp() <= @archived_timestamp() + if archived then return @archived_state() else return @archived_state() and @muted_state() + + @is_cleared = ko.computed => + return @last_event_timestamp() <= @cleared_timestamp() + + @is_verified = ko.computed => + return false if @participating_user_ets().length is 0 + return @participating_user_ets().every (user_et) -> user_et.is_verified() + + @removed_from_conversation = ko.observable false + @removed_from_conversation.subscribe (is_removed) => + @archived_state false if not is_removed + + ############################################################################### + # Messages + ############################################################################### + + @messages_unordered = ko.observableArray() + @messages = ko.computed => @messages_unordered().sort (a, b) -> a.timestamp - b.timestamp + @messages.subscribe => @update_latest_from_message @get_last_message() + + @creation_message = undefined + + @has_further_messages = ko.observable true + + @messages_visible = ko.pureComputed => + return [] if @id is '' + message_ets = (message_et for message_et in @messages() when message_et.visible()) + + first_message = @get_first_message() + if not @has_further_messages() and not (first_message?.is_member() and first_message.is_creation()) + @creation_message ?= @_creation_message() + message_ets.unshift @creation_message if @creation_message? + return message_ets + .extend trackArrayChanges: true + + # Call related + @call = ko.observable undefined + @has_active_call = ko.computed => + has_active_call = false + if @call()?.state() in z.calling.enum.CallStateGroups.IS_ACTIVE and not @call().is_ongoing_on_another_client() + has_active_call = true + return has_active_call + + @unread_events = ko.computed => + unread_event = [] + for message_et in @messages() when message_et.visible() by -1 + break if message_et.timestamp <= @last_read_timestamp() + unread_event.push message_et + return unread_event + + @number_of_unread_events = ko.computed => + return @unread_events().length + + @number_of_unread_messages = ko.computed => + return (message_et for message_et in @unread_events() when not message_et.user().is_me).length + + @unread_type = ko.computed => + return z.conversation.ConversationUnreadType.CONNECT if @connection().status() is z.user.ConnectionStatus.SENT + unread_type = z.conversation.ConversationUnreadType.UNREAD + return unread_type if @number_of_unread_messages() <= 0 + for message in @unread_events() by -1 + return z.conversation.ConversationUnreadType.MISSED_CALL if message.finished_reason is z.calling.enum.CallFinishedReason.MISSED + return z.conversation.ConversationUnreadType.PING if message.is_ping() + return unread_type + @unread_type.extend rateLimit: 50 + + ### + Display name strategy: + + @note 'One-to-One Conversations' and 'Connection Requests' + We should not use the conversation name received from the backend as fallback as it will always contain the + name of the user who received the connection request initially + + - Name of the other participant + - Name of the other user of the associated connection + - "..." if neither of those has been attached yet + + 'Group Conversation' + + - Conversation name received from backend + - If unnamed, we will create a name from the participant names + - Join the user's first names to a comma separated list or uses the user's first name if only one user participating + - "..." if the user entities have not yet been attached yet + ### + @display_name = ko.pureComputed + read: -> + if @type() in [z.conversation.ConversationType.CONNECT, z.conversation.ConversationType.ONE2ONE] + return @participating_user_ets()[0].name() if @participating_user_ets()[0]?.name() + return z.localization.Localizer.get_text z.string.truncation + else if @is_group() + return @name() if @name() + return (@participating_user_ets().map (user_et) -> user_et.first_name()).join ', ' if @participating_user_ets().length > 0 + return z.localization.Localizer.get_text z.string.conversation_list_empty_conversation if @participating_user_ids().length is 0 + return z.localization.Localizer.get_text z.string.truncation + else + return @name() + write: (value) -> return + owner: @ + + amplify.subscribe z.event.WebApp.CONVERSATION.LOADED_STATES, @_subscribe_to_states_updates + + _subscribe_to_states_updates: => + @archived_state.subscribe => + amplify.publish z.event.WebApp.CONVERSATION.STORE, @, z.conversation.ConversationUpdateType.ARCHIVED_STATE + @cleared_timestamp.subscribe => + amplify.publish z.event.WebApp.CONVERSATION.STORE, @, z.conversation.ConversationUpdateType.CLEARED_TIMESTAMP + @last_event_timestamp.subscribe => + amplify.publish z.event.WebApp.CONVERSATION.STORE, @, z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + @last_read_timestamp.subscribe => + amplify.publish z.event.WebApp.CONVERSATION.STORE, @, z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + @muted_state.subscribe => + amplify.publish z.event.WebApp.CONVERSATION.STORE, @, z.conversation.ConversationUpdateType.MUTED_STATE + + ############################################################################### + # Lifecycle + ############################################################################### + + # Remove all message from conversation unless there are unread events + release: => + if @number_of_unread_events() is 0 + @remove_messages() + @is_loaded false + @has_further_messages true + + ############################################################################### + # E2EE state setters + ############################################################################### + + ### + Set the timestamp of a given type. + + @note This will only increment timestamps + + @param timestamp [String] Timestamp to be set + @param type [z.conversation.ConversationUpdateType] Type of timestamp to be updated + @return [String] Timestamp value + ### + set_timestamp: (timestamp, type) => + switch type + when z.conversation.ConversationUpdateType.ARCHIVED_TIMESTAMP + entity_timestamp = @archived_timestamp + when z.conversation.ConversationUpdateType.CLEARED_TIMESTAMP + entity_timestamp = @cleared_timestamp + when z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + entity_timestamp = @last_event_timestamp + when z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + entity_timestamp = @last_read_timestamp + when z.conversation.ConversationUpdateType.MUTED_TIMESTAMP + entity_timestamp = @muted_timestamp + + updated_timestamp = @_increment_time_only entity_timestamp(), timestamp + if updated_timestamp + entity_timestamp updated_timestamp + return updated_timestamp + + ### + Increment only on timestamp update + + @param current_timestamp [z.entity.Conversation] Current timestamp + @param updated_timestamp [String] Timestamp from update + @return [String, Boolean] Updated timestamp or false if not increased + ### + _increment_time_only: (current_timestamp, updated_timestamp) -> + if updated_timestamp > current_timestamp then return updated_timestamp else return false + + ############################################################################### + # Messages + ############################################################################### + + ### + Adds a single message to the conversation. + @param message_et [z.entity.Message] Message entity to be added to the conversation + ### + add_message: (message_et) -> + @_update_last_read_from_message message_et + @messages_unordered.push @_check_for_duplicate_nonce message_et, @get_last_message() + + ### + Adds multiple messages to the conversation. + @param message_ets [z.entity.Message[]] Array of message entities to be added to the conversation + ### + add_messages: (message_ets) -> + for message_et, i in message_ets + message_et = @_check_for_duplicate_nonce message_ets[i - 1], message_et + + # in order to avoid multiple db writes check the messages from the end and stop once + # we found a message from self user + for message_et in message_ets by -1 + if message_et.user()?.is_me + @_update_last_read_from_message message_et + break + + z.util.ko_array_push_all @messages_unordered, message_ets + + ### + Prepends messages with new batch of messages. + @param message_ets [z.entity.Message[]] Array of messages to be added to conversation + ### + prepend_messages: (message_ets) -> + last_message_et = message_ets[message_ets.length - 1] + last_message_et = @_check_for_duplicate_nonce last_message_et, @get_first_message() + for message_et, i in message_ets by -1 + message_et = @_check_for_duplicate_nonce message_ets[i - 1], message_et + z.util.ko_array_unshift_all @messages_unordered, message_ets + + ### + Removes a single message from the conversation by message id. + @param message_id [String] ID of the message entity to be removed from the conversation + ### + remove_message_by_id: (message_id) -> + message_et = @get_message_by_id message_id + @messages_unordered.remove message_et if message_et? + + ### + Removes a single message from the conversation. + @param message_et [z.entity.Message] Message entity to be removed from the conversation + ### + remove_message: (message_et) -> + @messages_unordered.remove message_et + + ### + Removes all messages from the conversation. + ### + remove_messages: -> + @messages_unordered.removeAll() + + ### + Replace a message in the conversation. + + @param old_message_et [z.entity.Message] Message to be replaced + @param new_message_et [z.entity.Message] Message replacing the old one + ### + replace_message: (old_message_et, new_message_et) -> + @messages()[@messages.indexOf old_message_et] = new_message_et + @messages.valueHasMutated() + + ### + Checks for message duplicates by nonce and returns the message. + + @private + @note If a message is send to the backend multiple times by a client they will be in the conversation multiple times + + @param message_et [z.entity.Message] Message entity to be added to the conversation + @param other_message_et [z.entity.Message] Other message entity to compare with + ### + _check_for_duplicate_nonce: (message_et, other_message_et) -> + return message_et if not message_et? or not other_message_et? + if message_et.has_nonce() and other_message_et.has_nonce() and message_et.nonce is other_message_et.nonce + sorted_messages = z.entity.Message.sort_by_timestamp [message_et, other_message_et] + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.EVENT_HIDDEN_DUE_TO_DUPLICATE_NONCE + if message_et.type is z.event.Backend.CONVERSATION.ASSET_META and other_message_et.type is z.event.Backend.CONVERSATION.ASSET_META + # android sends to meta messages with the same content. we would store both and but only update the first one + # whenever we reload the conversation the nonce check would hide the older one and we would show the non updated message + # to fix that we hide the newer one + sorted_messages[1].visible false # hide newer + else + sorted_messages[0].visible false # hide older + return message_et + + + ############################################################################### + # Generated messages + ############################################################################### + + ### + Creates the placeholder message after clearing a conversation. + @note Only create the message if the group participants have been set + @private + ### + _creation_message: => + return undefined if @participating_user_ets().length is 0 + message_et = new z.entity.MemberMessage() + message_et.type = z.message.SuperType.MEMBER + message_et.user_ids @participating_user_ids() + message_et.user_ets @participating_user_ets().slice 0 + + if @type() in [z.conversation.ConversationType.CONNECT, z.conversation.ConversationType.ONE2ONE] + if @participating_user_ets()[0].sent() + message_et.member_message_type = z.message.SystemMessageType.CONNECTION_REQUEST + else + message_et.member_message_type = z.message.SystemMessageType.CONNECTION_ACCEPTED + else + message_et.member_message_type = z.message.SystemMessageType.CONVERSATION_CREATE + if @creator is @self.id + message_et.user @self + else + message_et.user_ets.push @self + user_et = ko.utils.arrayFirst @participating_user_ets(), (user_et) => + return user_et.id is @creator + if user_et + message_et.user user_et + else + message_et.member_message_type = z.message.SystemMessageType.CONVERSATION_RESUME + return message_et + + ### + Creates a E2EE message of type z.message.E2EEMessageType.ALL_VERIFIED. + @private + ### + _verified_message: -> + message_et = new z.entity.Message() + message_et.type = z.message.SuperType.ALL_VERIFIED + return message_et + + ### + Creates a E2EE message of type z.message.E2EEMessageType.ALL_VERIFIED. + @private + ### + _new_device_message: -> + message_et = new z.entity.DeviceMessage() + return message_et + + ### + Creates a E2EE message of type z.message.E2EEMessageType.ALL_VERIFIED. + @private + ### + _unverified_device_message: -> + message_et = new z.entity.DeviceMessage() + message_et.unverified true + return message_et + + + ############################################################################### + # Update last activity + ############################################################################### + + ### + Update information about last activity from multiple messages. + @param message_ets [z.entity.Message[]] Array of messages to be added to conversation + ### + update_latest_from_messages: (message_ets) -> + last_message = message_ets[message_ets.length - 1] + @update_latest_from_message last_message + + ### + Update information about last activity from single message. + @param message_et [z.entity.Message] Message to be added to conversation + ### + update_latest_from_message: (message_et) -> + if message_et? and message_et.visible() + @set_timestamp message_et.timestamp, z.conversation.ConversationUpdateType.LAST_EVENT_TIMESTAMP + + ### + Update last read if message sender is self + @private + @param message_et [z.entity.Message] + ### + _update_last_read_from_message: (message_et) -> + if message_et.user()?.is_me and message_et.timestamp + @set_timestamp message_et.timestamp, z.conversation.ConversationUpdateType.LAST_READ_TIMESTAMP + + ############################################################################### + # Get messages + ############################################################################### + + ### + Get all messages. + @return [z.entity.Message[ko.observableArray]] Array of all message in the conversation + ### + get_all_messages: -> + return @messages() + + ### + Returns a message with an image if found by correlation ID and image type. + + @param message_et [z.entity.Message] Message with image for which a correlating message should be found + @param message_ets [Array] Pool of message to search in first + @return [z.entity.ContentMessage] Correlating content message containing image + ### + get_correlating_image_message: (message_et, message_ets) => + image_message_et = @_get_correlating_image_message message_et, message_ets if message_ets?.length > 0 + image_message_et = @_get_correlating_image_message message_et, @messages() if not image_message_et? + return image_message_et + + ### + Get the first message of the conversation. + @return [z.entity.Message, undefined] First message entity or undefined + ### + get_first_message: -> + return @messages()[0] + + ### + Get the last message of the conversation. + @return [z.entity.Message, undefined] Last message entity or undefined + ### + get_last_message: -> + return @messages()[@messages().length - 1] + + ### + Get a message by it's unique ID. + @param id [String] ID of message to be retrieved + @return [z.entity.Message, undefined] Message with ID or undefined + ### + get_message_by_id: (id) -> + for message_et in @messages() + return message_et if message_et.id is id + + ### + Returns a message with an image if found by correlation ID and image type. + + @private + @param message_et [z.entity.Message] Message with image for which a correlating message should be found + @param message_ets [Array] Pool of message to search in + @return [z.entity.ContentMessage] Correlating content message containing image + ### + _get_correlating_image_message: (input_message_et, message_ets) -> + input_asset_et = input_message_et.get_first_asset() + for other_message_et in message_ets when other_message_et.has_asset_image() + continue if other_message_et.id is input_message_et.id + other_asset_et = other_message_et.get_first_asset() + if other_asset_et.correlation_id is input_asset_et.correlation_id + return other_message_et if input_asset_et.is_medium_image() is other_asset_et.is_preview_image() + + ### + Get Number of pending uploads for this conversation. + ### + get_number_of_pending_uploads: -> + pending_uploads = (message_et for message_et in @messages() when message_et.assets?()[0]?.pending_upload?()) + return pending_uploads.length + + ############################################################################### + # Serialization + ############################################################################### + + serialize: => + return {} = + id: @id + archived_state: @archived_state() + archived_timestamp: @archived_timestamp() + cleared_timestamp: @cleared_timestamp() + last_event_timestamp: @last_event_timestamp() + last_read_timestamp: @last_read_timestamp() + muted_state: @muted_state() + muted_timestamp: @muted_timestamp() diff --git a/app/script/entity/User.coffee b/app/script/entity/User.coffee new file mode 100644 index 00000000000..b359180ebb0 --- /dev/null +++ b/app/script/entity/User.coffee @@ -0,0 +1,140 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# User entity. +# Please note: The own user has a "tracking_id" & "locale" +class z.entity.User + + + THEME: + BLUE: 'theme-blue' + GREEN: 'theme-green' + ORANGE: 'theme-orange' + PINK: 'theme-pink' + PURPLE: 'theme-purple' + RED: 'theme-red' + YELLOW: 'theme-yellow' + + # TODO remove (don't hardcode color values), set and handle this in CSS classes + ACCENT_COLOR: + BLUE: '#2391d3' + GREEN: '#00c800' + ORANGE: '#ff8900' + PINK: '#fe5ebd' + PURPLE: '#9c00fe' + RED: '#fb0807' + YELLOW: '#febf02' + + ### + Construct a new user entity. + + @param user_id [String] User ID + ### + constructor: (user_id = '') -> + @id = user_id + @is_me = false + + @joaat_hash = -1 + + @accent_id = ko.observable z.config.ACCENT_ID.BLUE + @accent_theme = ko.computed => + switch @accent_id() + when z.config.ACCENT_ID.BLUE then return @THEME.BLUE + when z.config.ACCENT_ID.GREEN then return @THEME.GREEN + when z.config.ACCENT_ID.ORANGE then return @THEME.ORANGE + when z.config.ACCENT_ID.PINK then return @THEME.PINK + when z.config.ACCENT_ID.PURPLE then return @THEME.PURPLE + when z.config.ACCENT_ID.RED then return @THEME.RED + when z.config.ACCENT_ID.YELLOW then return @THEME.YELLOW + else return @THEME.BLUE + , @, deferEvaluation: true + + @accent_color = ko.computed => + switch @accent_id() + when z.config.ACCENT_ID.BLUE then return @ACCENT_COLOR.BLUE + when z.config.ACCENT_ID.GREEN then return @ACCENT_COLOR.GREEN + when z.config.ACCENT_ID.ORANGE then return @ACCENT_COLOR.ORANGE + when z.config.ACCENT_ID.PINK then return @ACCENT_COLOR.PINK + when z.config.ACCENT_ID.PURPLE then return @ACCENT_COLOR.PURPLE + when z.config.ACCENT_ID.RED then return @ACCENT_COLOR.RED + when z.config.ACCENT_ID.YELLOW then return @ACCENT_COLOR.YELLOW + else return @ACCENT_COLOR.BLUE + , @, deferEvaluation: true + + @phone = ko.observable() + @display_phone = ko.computed => + formatted_number = PhoneFormat.formatNumberForMobileDialing '', @phone() + return formatted_number or @phone() + , deferEvaluation: true + + @email = ko.observable() + + @name = ko.observable '' + @first_name = ko.computed => + return @name().split(' ')[0] + @last_name = ko.computed => + parts = @name().split(' ') + return parts.pop() if parts.length > 1 + @initials = ko.computed => + initials = '' + if @first_name()? and @last_name()? + initials = z.util.get_first_character(@first_name()) + z.util.get_first_character(@last_name()) + else + initials = @first_name().slice 0, 2 + return initials.toUpperCase() + + @mutual_friend_ets = ko.observableArray [] + @mutual_friend_ids = ko.observableArray [] + @mutual_friends_total = ko.observable 0 + + @picture_preview = ko.observable '' + @picture_medium = ko.observable '' + @raw_pictures = ko.observable [] + + # TODO: Quickfix for picture previews. Tiago thinks about a better solution! + # TODO: Make sure that this function returns only an URL (without url('...')) + @picture_preview_url = => + return @picture_preview() if @picture_preview().length is 0 + url = @picture_preview().substr(0, @picture_preview().indexOf('?')) + return "#{url}?access_token=#{wire.auth.client.access_token}&conv_id=#{@id}')" + + @picture_medium_url = => + return @picture_medium() if @picture_medium().length is 0 + url = @picture_medium().substr(0, @picture_medium().indexOf('?')) + return "#{url}?access_token=#{wire.auth.client.access_token}&conv_id=#{@id}')" + + @connection_level = ko.observable z.user.ConnectionLevel.UNKNOWN + @connection = ko.observable new z.entity.Connection() + + # connection state shorthands TODO add others too since this is used very often? + @blocked = ko.computed => @connection().status() is z.user.ConnectionStatus.BLOCKED + @connected = ko.computed => @connection().status() is z.user.ConnectionStatus.ACCEPTED + @sent = ko.computed => @connection().status() is z.user.ConnectionStatus.SENT + + # e2ee + @devices = ko.observableArray() + @is_verified = ko.computed => + return false if @devices().length is 0 + return @devices().every (client_et) -> client_et.meta.is_verified() + + add_client: (client_et) -> + is_existent = ko.utils.arrayFirst @devices(), (exisiting_client) -> exisiting_client.id is client_et.id + @devices.push client_et if not is_existent diff --git a/app/script/entity/message/Asset.coffee b/app/script/entity/message/Asset.coffee new file mode 100644 index 00000000000..968c124be02 --- /dev/null +++ b/app/script/entity/message/Asset.coffee @@ -0,0 +1,121 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Asset entity. +class z.entity.Asset + + ### + Construct a new asset. + + @param id [String] Asset ID + ### + constructor: (@id) -> + @key = '' + @type = '' + + ### + Check if asset is a medium image. + + @return [Boolean] Is asset of type medium image + ### + is_medium_image: -> + return @type is z.assets.AssetType.MEDIUM_IMAGE + + ### + Check if asset is a preview image. + + @return [Boolean] Is asset of type preview image + ### + is_preview_image: -> + return @type is z.assets.AssetType.PREVIEW_IMAGE + + ### + Check if asset is a text. + + @return [Boolean] Is asset of type text + ### + is_text: -> + return @type is z.assets.AssetType.TEXT + + ### + Check if asset is a file. + + @return [Boolean] Is asset of type file + ### + is_file: -> + return @type is z.assets.AssetType.FILE and not @is_video() and not @is_audio() + + ### + Check if asset is a location. + + @return [Boolean] Is asset of type location + ### + is_location: -> + return @type is z.assets.AssetType.LOCATION + + ### + Check if asset is a video. + + @return [Boolean] Is asset of type video + ### + is_video: -> + if @type is z.assets.AssetType.FILE and @file_type?.startsWith('video') and not z.util.Environment.browser.firefox + can_play_audio = document.createElement('video').canPlayType @file_type + return true if can_play_audio isnt '' + return false + ### + Check if asset is a audio. + + @return [Boolean] Is asset of type audio + ### + is_audio: -> + if @type is z.assets.AssetType.FILE and @file_type?.startsWith 'audio' + can_play_audio = document.createElement('audio').canPlayType @file_type + return true if can_play_audio isnt '' + return false + + ### + Replace access token in image url. + + @param url [String] asset url + @param access_token [String] + @return url [String] + ### + _replace_access_token: (url, access_token) -> + token = url.match /access_token=([^&\s]*)/i + return url.replace token[1], access_token + + ### + Get current asset url. + + @param url [String] asset url + @return url [String] + ### + generate_asset_url: (url) => + @_replace_access_token url, wire.auth.client.access_token + + ### + Process asset before rendering it + + @todo Implement + ### + render: -> + return 'Not implemented' diff --git a/app/script/entity/message/CallMessage.coffee b/app/script/entity/message/CallMessage.coffee new file mode 100644 index 00000000000..0d090fb0172 --- /dev/null +++ b/app/script/entity/message/CallMessage.coffee @@ -0,0 +1,50 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Call message entity based on z.entity.Message. +class z.entity.CallMessage extends z.entity.Message + # Construct a new content message. + constructor: -> + super() + @super_type = z.message.SuperType.CALL + @call_message_type = '' + @finished_reason = '' + + @caption = ko.computed => + return z.localization.Localizer.get_text z.string.conversation_voice_channel_deactivate_you if @user().is_me + return z.localization.Localizer.get_text z.string.conversation_voice_channel_deactivate + , @, deferEvaluation: true + + ### + Check if call message is call activation. + + @return [Boolean] Is message of type activate + ### + is_call_activation: -> + return @call_message_type is z.message.CallMessageType.ACTIVATED + + ### + Check if call message is call activation. + + @return [Boolean] Is message of type deactivate + ### + is_call_deactivation: -> + return @call_message_type is z.message.CallMessageType.DEACTIVATED diff --git a/app/script/entity/message/ContentMessage.coffee b/app/script/entity/message/ContentMessage.coffee new file mode 100644 index 00000000000..0984c4f0560 --- /dev/null +++ b/app/script/entity/message/ContentMessage.coffee @@ -0,0 +1,80 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Content Message based on z.entity.Message. +class z.entity.ContentMessage extends z.entity.Message + # Construct a new content message. + constructor: -> + super() + + @assets = ko.observableArray [] + @nonce = null + @super_type = z.message.SuperType.CONTENT + + + ### + Add another content asset to the message. + + @param asset_et [z.entity.Asset] New content asset + ### + add_asset: (asset_et) => + @assets.push asset_et + + ### + Get the first asset attached to the message. + + @return [z.entity.Message] The first asset attached to the message + ### + get_first_asset: -> + return @assets()[0] + + ### + Gets the first asset attached to the message of a specified asset type. + + @param asset_type [z.assets.AssetType] Type the asset should be of + + @return [z.entity.Asset] First matching asset + ### + get_first_asset_of_type: (asset_type) -> + return asset_et for asset_et in @assets() when asset_et.type is asset_type + + ### + Gets the first asset attached to the message of a specified image type. + + @param image_type [z.assets.ImageType] Type the image asset should be of + + @return [z.entity.Asset] First matching asset + ### + get_first_image_of_type: (image_type) -> + if image_type is z.assets.ImageSizeType.MEDIUM + return asset_et for asset_et in @assets() when asset_et.type is z.asset.AssetType.MEDIUM_IMAGE + else if image_type is z.assets.ImageSizeType.PREVIEW + return asset_et for asset_et in @assets() when asset_et.type is z.asset.AssetType.PREVIEW_IMAGE + + ### + Check if message can be deleted. + + @return [Boolean] + ### + is_deletable: -> + asset = @assets()[0] + return true if not asset.is_file() + return asset.status() not in [z.assets.AssetTransferState.DOWNLOADING, z.assets.AssetTransferState.UPLOADING] diff --git a/app/script/entity/message/DecryptErrorMessage.coffee b/app/script/entity/message/DecryptErrorMessage.coffee new file mode 100644 index 00000000000..26b6ebdd6a6 --- /dev/null +++ b/app/script/entity/message/DecryptErrorMessage.coffee @@ -0,0 +1,69 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +class z.entity.DecryptErrorMessage extends z.entity.Message + @::RECOVERABLE_STATES = [ + z.cryptography.CryptographyErrorType.DUPLICATE_MESSAGE + z.cryptography.CryptographyErrorType.INVALID_MESSAGE_SESSION_NOT_MATCHING + z.cryptography.CryptographyErrorType.INVALID_MESSAGE_SESSION_MISSING + z.cryptography.CryptographyErrorType.INVALID_SIGNATURE + z.cryptography.CryptographyErrorType.OUTDATED_MESSAGE + z.cryptography.CryptographyErrorType.PRE_KEY_NOT_FOUND + z.cryptography.CryptographyErrorType.TOO_DISTANT_FUTURE + ] + + constructor: -> + super() + @super_type = z.message.SuperType.UNABLE_TO_DECRYPT + + @error_code = '' + @client_id = '' + + @caption = ko.pureComputed => + caption_id = z.string.conversation_unable_to_decrypt_1 + caption_id = z.string.conversation_unable_to_decrypt_2 if @error_code is z.cryptography.CryptographyErrorType.REMOTE_IDENTITY_CHANGED + return z.localization.Localizer.get_text { + id: caption_id + replace: { + placeholder: '%@name', content: "#{ z.util.escape_html @user().first_name()}" + } + } + + @link = ko.pureComputed => + return z.localization.Localizer.get_text z.string.url_decrypt_error_2 if @error_code is '3690' + return z.localization.Localizer.get_text z.string.url_decrypt_error_1 + + @is_recoverable = ko.pureComputed => + return @error_code in @RECOVERABLE_STATES + + @is_resetting_session = ko.observable false + + @error_message = ko.pureComputed => + parts = [] + + if @error_code + error_text = z.localization.Localizer.get_text z.string.conversation_unable_to_decrypt_error_message + parts.push "#{error_text}: #{@error_code}" + + if @client_id + parts.push "ID: #{z.util.print_devices_id(@client_id)}" + + return "(#{parts.join(' ')})" if parts.length diff --git a/app/script/entity/message/File.coffee b/app/script/entity/message/File.coffee new file mode 100644 index 00000000000..2f631249714 --- /dev/null +++ b/app/script/entity/message/File.coffee @@ -0,0 +1,128 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Medium image asset entity. +class z.entity.File extends z.entity.Asset + ### + Construct a new medium image asset. + + @param id [String] Asset ID + ### + constructor: (id) -> + super id + @type = z.assets.AssetType.FILE + @logger = new z.util.Logger 'z.entity.File', z.config.LOGGER.OPTIONS + + # z.assets.AssetTransferState + @status = ko.observable() + + @file_name = '' + @file_size = '' + @file_type = '' + + # contains asset meta data as object + @meta = {} + + # asset url, instance of an otr asset this has to be decrypted + @original_resource = ko.observable() + @preview_resource = ko.observable() + + @download_progress = ko.pureComputed => @original_resource()?.download_progress() + @cancel_download = => @original_resource()?.cancel_download() + + @upload_id = ko.observable() + @upload_progress = ko.observable() + @uploaded_on_this_client = ko.observable false + @upload_failed_reason = ko.observable() + @pending_upload = ko.pureComputed => + return @status() is z.assets.AssetTransferState.UPLOADING and @uploaded_on_this_client() + + # update progress + @upload_id.subscribe (id) => + amplify.subscribe 'upload' + id, @on_progress if id + + @status.subscribe (status) => + if status is z.assets.AssetTransferState.UPLOADED + amplify.unsubscribe 'upload' + @upload_id, @on_progress + + on_progress: (progress) => + @upload_progress progress + + ### + Loads and decrypts otr asset preview + + @return [Promise] Returns a promise that resolves with the asset as blob + ### + load_preview: => + @preview_resource()?.load() + + ### + Loads and decrypts otr asset + + @return [Promise] Returns a promise that resolves with the asset as blob + ### + load: => + @status z.assets.AssetTransferState.DOWNLOADING + @original_resource()?.load() + .then (blob) => + @status z.assets.AssetTransferState.UPLOADED + return blob + .catch (error) => + @status z.assets.AssetTransferState.UPLOADED + throw error + + ### + Loads and decrypts otr asset as initiates download + + @return [Promise] Returns a promise that resolves with the asset as blob + ### + download: => + return if @status() isnt z.assets.AssetTransferState.UPLOADED + + download_started = Date.now() + tracking_data = + size_bytes: @file_size + size_mb: z.util.bucket_values (@file_size / 1024 / 1024), [0, 5, 10, 15, 20, 25] + type: z.util.get_file_extension @file_name + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.DOWNLOAD_INITIATED, tracking_data + + @load() + .then (blob) => + return z.util.download_blob blob, @file_name + .then => + download_duration = (Date.now() - download_started) / 1000 + @logger.log "Downloaded asset in #{download_duration} seconds" + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.DOWNLOAD_SUCCESSFUL, + $.extend tracking_data, {time: download_duration} + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to download asset', error + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.DOWNLOAD_FAILED, tracking_data + + cancel: (message_et) => + amplify.publish z.event.WebApp.CONVERSATION.ASSET.CANCEL, message_et + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.UPLOAD_CANCELLED, + size_bytes: @file_size + size_mb: z.util.bucket_values (@file_size / 1024 / 1024), [0, 5, 10, 15, 20, 25] + type: z.util.get_file_extension @file_name + + reload: => + @logger.log 'Restart upload' diff --git a/app/script/entity/message/LinkPreview.coffee b/app/script/entity/message/LinkPreview.coffee new file mode 100644 index 00000000000..6f1903be310 --- /dev/null +++ b/app/script/entity/message/LinkPreview.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +class z.entity.LinkPreview + constructor: -> + + @title = '' + @summary = '' + @orginal_url = '' + @permanent_url = '' + @url_offset = 0 + + # z.assets.AssetRemoteData + @image_resource = ko.observable() diff --git a/app/script/entity/message/Location.coffee b/app/script/entity/message/Location.coffee new file mode 100644 index 00000000000..dfa82ab0e20 --- /dev/null +++ b/app/script/entity/message/Location.coffee @@ -0,0 +1,33 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +class z.entity.Location extends z.entity.Asset + constructor: -> + super() + @type = z.assets.AssetType.LOCATION + + @latitude = '' + @longitude = '' + @name = '' + @zoom = '' + + @link_src = ko.pureComputed => + z.location.get_maps_url @latitude, @longitude, @name, @zoom diff --git a/app/script/entity/message/MediumImage.coffee b/app/script/entity/message/MediumImage.coffee new file mode 100644 index 00000000000..f3c6bca39d9 --- /dev/null +++ b/app/script/entity/message/MediumImage.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Medium image asset entity. +class z.entity.MediumImage extends z.entity.Asset + ### + Construct a new medium image asset. + + @param id [String] Asset ID + ### + constructor: (id) -> + super id + @type = z.assets.AssetType.MEDIUM_IMAGE + + @correlation_id = '' + + @width = '0px' + @height = '0px' + + # z.assets.AssetRemoteData + @resource = ko.observable() diff --git a/app/script/entity/message/MemberMessage.coffee b/app/script/entity/message/MemberMessage.coffee new file mode 100644 index 00000000000..f686166873d --- /dev/null +++ b/app/script/entity/message/MemberMessage.coffee @@ -0,0 +1,111 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +### +Member message entity based on z.entity.SystemMessage. + +@todo Refactor for a proper SystemMessage entities +### +class z.entity.MemberMessage extends z.entity.SystemMessage + # Construct a new member message. + constructor: -> + super() + @super_type = z.message.SuperType.MEMBER + @member_message_type = z.message.SystemMessageType.NORMAL + + @user_ets = ko.observableArray() + @user_ids = ko.observableArray() + + # Users joined the conversation without sender + @joined_user_ets = ko.pureComputed => + return (user_et for user_et in @user_ets() when user_et.id isnt @user().id) + + # Users joined the conversation without self + @remote_user_ets = ko.pureComputed => + return (user_et for user_et in @user_ets() when not user_et.is_me) + + @_generate_name_string = (declension = z.string.Declension.ACCUSATIVE) => + names_string = (z.util.get_first_name user_et, declension for user_et in @joined_user_ets()).join ', ' + return names_string.replace /,(?=[^,]*$)/, " #{z.localization.Localizer.get_text z.string.and}" + + @_get_caption_connection = (connection_status) -> + switch connection_status + when z.user.ConnectionStatus.BLOCKED + identifier = z.string.conversation_connection_blocked + when z.user.ConnectionStatus.SENT then return '' + else + identifier = z.string.conversation_connection_accepted + return z.localization.Localizer.get_text identifier + + @_get_caption_with_names = (key, declension) => + return z.localization.Localizer.get_text { + id: key + replace: {placeholder: '%@names', content: @_generate_name_string declension} + } + + @show_large_avatar = => + large_avatar_types = [ + z.message.SystemMessageType.CONNECTION_ACCEPTED + z.message.SystemMessageType.CONNECTION_REQUEST + ] + return @member_message_type in large_avatar_types + + @other_user = ko.computed => + if @user_ets().length is 1 then @user_ets()[0] else new z.entity.User() + + @caption = ko.computed => + return '' if @user_ets().length is 0 + + switch @member_message_type + when z.message.SystemMessageType.CONNECTION_ACCEPTED, z.message.SystemMessageType.CONNECTION_REQUEST + return @_get_caption_connection @other_user().connection().status() + when z.message.SystemMessageType.CONVERSATION_CREATE + return @_get_caption_with_names z.string.conversation_create_you if @user().is_me + return @_get_caption_with_names z.string.conversation_create, z.string.Declension.DATIVE + when z.message.SystemMessageType.CONVERSATION_RESUME + return @_get_caption_with_names z.string.conversation_resume, z.string.Declension.DATIVE + + switch @type + when z.event.Backend.CONVERSATION.MEMBER_LEAVE + if @other_user().id is @user().id + return z.localization.Localizer.get_text z.string.conversation_member_leave_left_you if @user().is_me + return z.localization.Localizer.get_text z.string.conversation_member_leave_left + return @_get_caption_with_names z.string.conversation_member_leave_removed_you if @user().is_me + return @_get_caption_with_names z.string.conversation_member_leave_removed + when z.event.Backend.CONVERSATION.MEMBER_JOIN + return @_get_caption_with_names z.string.conversation_member_join_you if @user().is_me + return @_get_caption_with_names z.string.conversation_member_join + + , @, deferEvaluation: true + + is_connection: => + return @member_message_type in [ + z.message.SystemMessageType.CONNECTION_ACCEPTED + z.message.SystemMessageType.CONNECTION_REQUEST + ] + + is_creation: => + return @member_message_type in [ + z.message.SystemMessageType.CONNECTION_ACCEPTED + z.message.SystemMessageType.CONNECTION_REQUEST + z.message.SystemMessageType.CONVERSATION_CREATE + z.message.SystemMessageType.CONVERSATION_RESUME + ] diff --git a/app/script/entity/message/Message.coffee b/app/script/entity/message/Message.coffee new file mode 100644 index 00000000000..bfc79e32a84 --- /dev/null +++ b/app/script/entity/message/Message.coffee @@ -0,0 +1,209 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Base message entity. +class z.entity.Message + # Construct a new base message entity. + constructor: -> + @from = '' + @id = '0' + @primary_key = undefined + @super_type = '' + @timestamp = Date.now() + @type = '' + @user = ko.observable new z.entity.User() + @visible = ko.observable true + + @display_timestamp_short = => + date = moment.unix @timestamp / 1000 + return date.local().format 'HH:mm' + + @sender_name = ko.computed => + z.util.get_first_name @user() + , @, deferEvaluation: true + + @accent_color = ko.computed => + return "accent-color-#{@user().accent_id()}" + , @, deferEvaluation: true + + ### + Check if message contains a preview image asset. + + @return [Boolean] Message contains a preview image + ### + has_asset: -> + if @is_content() + return true for asset_et in @assets() when asset_et.type is z.assets.AssetType.FILE + return false + + ### + Check if message contains any image asset. + + @return [Boolean] Message contains any image + ### + has_asset_image: -> + if @is_content() + return true for asset_et in @assets() when asset_et.is_medium_image() or asset_et.is_preview_image() + return false + + ### + Check if message contains a medium image asset. + + @return [Boolean] Message contains a medium image + ### + has_asset_medium_image: -> + if @is_content() + return true for asset_et in @assets() when asset_et.is_medium_image() + return false + + ### + Check if message contains a preview image asset. + + @return [Boolean] Message contains a preview image + ### + has_asset_preview_image: -> + if @is_content() + return true for asset_et in @assets() when asset_et.is_preview_image() + return false + + ### + Check if message contains a preview image asset. + + @return [Boolean] Message contains a preview image + ### + has_asset_file: -> + if @is_content() + return true for asset_et in @assets() when asset_et.is_file() + return false + + ### + Check if message contains a text asset. + + @return [Boolean] Message contains text + ### + has_asset_text: -> + if @is_content() + return true for asset_et in @assets() when asset_et.is_text() + return false + + ### + Check if message contains a nonce. + + @return [Boolean] Message contains a nonce + ### + has_nonce: -> + return @super_type in [z.message.SuperType.CONTENT] + + ### + Check if message is a call message. + + @return [Boolean] Is message of type call + ### + is_call: -> + return @super_type is z.message.SuperType.CALL + + ### + Check if message is a content message. + + @return [Boolean] Is message of type content + ### + is_content: -> + return @super_type is z.message.SuperType.CONTENT + + ### + Check if message is a member message. + + @return [Boolean] Is message of type member + ### + is_member: -> + return @super_type is z.message.SuperType.MEMBER + + ### + Check if message is a ping message. + + @return [Boolean] Is message of type ping + ### + is_ping: -> + return @super_type is z.message.SuperType.PING + + ### + Check if message is a system message. + + @return [Boolean] Is message of type system + ### + is_system: -> + return @super_type is z.message.SuperType.SYSTEM + + ### + Check if message is a e2ee message. + + @return [Boolean] Is message of type system + ### + is_device: -> + return @super_type is z.message.SuperType.DEVICE + + ### + Check if message is a e2ee message. + + @return [Boolean] Is message of type system + ### + is_all_verified: -> + return @super_type is z.message.SuperType.ALL_VERIFIED + + ### + Check if message is a e2ee message. + + @return [Boolean] Is message of type system + ### + is_unable_to_decrypt: -> + return @super_type is z.message.SuperType.UNABLE_TO_DECRYPT + + ### + Triggers event to delete message. + ### + delete: => + active_conversation = wire.app.repository.conversation.active_conversation() + type = 'text' + + if @is_system() + type = 'system' + else if @is_ping() + type = 'ping' + else if @has_asset_image() + type = 'image' + else if @has_asset() + type = 'file' + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.SELECTED_MESSAGE, + context: 'single' + conversation_type: if active_conversation.is_one2one() then 'one_to_one' else 'group' + type: type + + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.DELETE_MESSAGE, + action: => amplify.publish z.event.WebApp.CONVERSATION.MESSAGE.DELETE, @ + + ### + Sort messages by timestamp + + @return [Boolean] Is message of type system + ### + @sort_by_timestamp: (message_ets) -> + message_ets.sort (m1, m2) -> m1.timestamp > m2.timestamp diff --git a/app/script/entity/message/NewDeviceMessage.coffee b/app/script/entity/message/NewDeviceMessage.coffee new file mode 100644 index 00000000000..e6f5670135b --- /dev/null +++ b/app/script/entity/message/NewDeviceMessage.coffee @@ -0,0 +1,61 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# E2EE new device message entity based on z.entity.Message. +class z.entity.NewDeviceMessage extends z.entity.Message + # Construct a new content message. + constructor: -> + super() + @type = z.message.SuperType.NEW_DEVICE + @device = ko.observable() + @device_owner = ko.observable new z.entity.User() + + @unverified = ko.observable false + + # TODO + # You started using this device -> settings + # You started using a new device -> settings + # John started using a new device -> profile + # You unverified one of John's devices -> profile + # You unverified one of your devices -> settings + + @caption = ko.computed => + return z.localization.Localizer.get_text z.string.conversation_device_unverified if @unverified() + return z.localization.Localizer.get_text z.string.conversation_device_started_using_you if @device_owner().is_me + return z.localization.Localizer.get_text z.string.conversation_device_started_using + + @caption_device = ko.computed => + if @unverified() + return z.localization.Localizer.get_text z.string.conversation_device_your_devices if @device_owner().is_me + return z.localization.Localizer.get_text { + id: z.string.conversation_device_user_devices + replace: {placeholder: '%@name', content: @device_owner().first_name()} + } + else + # TODO current device + return z.localization.Localizer.get_text z.string.conversation_device_a_new_device + + click_on_device: => + # TODO device + if @device_owner()?.is_me + amplify.publish z.event.WebApp.PROFILE.SETTINGS.SHOW + else + amplify.subscribe z.event.WebApp.SHORTCUT.PEOPLE diff --git a/app/script/entity/message/PingMessage.coffee b/app/script/entity/message/PingMessage.coffee new file mode 100644 index 00000000000..1665edf1bc5 --- /dev/null +++ b/app/script/entity/message/PingMessage.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +class z.entity.PingMessage extends z.entity.Message + constructor: -> + super() + @super_type = z.message.SuperType.PING + @animated = ko.observable false + + @caption = ko.computed => + string = if @user().is_me then z.string.conversation_ping_you else z.string.conversation_ping + return z.localization.Localizer.get_text string + , @, deferEvaluation: true + + @animation = ko.computed -> + return 'ping-animation ping-animation-soft' + , @, deferEvaluation: true diff --git a/app/script/entity/message/PreviewImage.coffee b/app/script/entity/message/PreviewImage.coffee new file mode 100644 index 00000000000..133ee7b4f35 --- /dev/null +++ b/app/script/entity/message/PreviewImage.coffee @@ -0,0 +1,54 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Preview image asset entity. +class z.entity.PreviewImage extends z.entity.Asset + ### + Construct a new medium image asset. + + @param id [String] Asset ID + ### + constructor: (id) -> + super id + @type = z.assets.AssetType.PREVIEW_IMAGE + + @correlation_id = '' + + @content_type = '' + @encoded_data = '' + + @width = '0px' + @height = '0px' + + # Process preview image before rendering it. + render: -> + src = "data:#{@content_type};base64,#{@encoded_data}" + url = "url(#{src})" + + config = + id: "preview-image-#{@correlation_id}" + width: window.parseInt @width, 10 + height: window.parseInt @height, 10 + preview_url: src + + image = new zeta.webapp.module.image.animation.ImageBuilder().build config + zeta.webapp.module.image.animation.onresize() + return image.clone().wrap('

    ').parent().html() diff --git a/app/script/entity/message/RenameMessage.coffee b/app/script/entity/message/RenameMessage.coffee new file mode 100644 index 00000000000..68e21875a7a --- /dev/null +++ b/app/script/entity/message/RenameMessage.coffee @@ -0,0 +1,30 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Rename message entity based on z.entity.SystemMessage. +class z.entity.RenameMessage extends z.entity.SystemMessage + # Construct a new system message. + constructor: -> + super() + @system_message_type = z.message.SystemMessageType.CONVERSATION_RENAME + @caption = ko.computed => + return z.localization.Localizer.get_text z.string.conversation_rename_you if @user().is_me + return z.localization.Localizer.get_text z.string.conversation_rename diff --git a/app/script/entity/message/SystemMessage.coffee b/app/script/entity/message/SystemMessage.coffee new file mode 100644 index 00000000000..ddb8d1f2409 --- /dev/null +++ b/app/script/entity/message/SystemMessage.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# System message entity based on z.entity.Message. +class z.entity.SystemMessage extends z.entity.Message + # Construct a new system message. + constructor: -> + super() + @super_type = z.message.SuperType.SYSTEM diff --git a/app/script/entity/message/Text.coffee b/app/script/entity/message/Text.coffee new file mode 100644 index 00000000000..edf367a7d19 --- /dev/null +++ b/app/script/entity/message/Text.coffee @@ -0,0 +1,52 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.entity ?= {} + +# Text asset entity. +class z.entity.Text extends z.entity.Asset + ### + Construct a new text asset. + + @param id [String] Asset ID + ### + constructor: (id) -> + super(id) + @type = z.assets.AssetType.TEXT + + # Raw message text + @text = '' + + # Can be used to theme media embeds + @theme_color = undefined + + # Processed text including media embeds, link highlight and markdown + @processed_text = undefined + + # Array of z.entity.LinkPreview instances + @previews = ko.observableArray() + + @should_render_text = ko.pureComputed => + has_link_previews = @previews().length > 0 + return not has_link_previews or (has_link_previews and not z.links.LinkPreviewHelpers.contains_only_link(@text)) + + # Process text before rendering it. + render: -> + @processed_text ?= z.util.render_message @text, @theme_color + return @processed_text diff --git a/app/script/event/Backend.coffee b/app/script/event/Backend.coffee new file mode 100644 index 00000000000..4050ef86b8e --- /dev/null +++ b/app/script/event/Backend.coffee @@ -0,0 +1,63 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +### +Enum of different backend events. +### +z.event.Backend = + CALL: + STATE: 'call.state' + FLOW_ADD: 'call.flow-add' + REMOTE_SDP: 'call.remote-sdp' + REMOTE_CANDIDATES_ADD: 'call.remote-candidates-add' + REMOTE_CANDIDATES_UPDATE: 'call.remote-candidates-update' + FLOW_ACTIVE: 'call.flow-active' + FLOW_DELETE: 'call.flow-delete' + CONVERSATION: + ASSET_ADD: 'conversation.asset-add' + ASSET_META: 'conversation.asset-meta' + ASSET_PREVIEW: 'conversation.asset-preview' + ASSET_UPLOAD_COMPLETE: 'conversation.asset-upload-complete' + ASSET_UPLOAD_FAILED: 'conversation.asset-upload-failed' + CLIENT_MESSAGE_ADD: 'conversation.client-message-add' + CONNECT_REQUEST: 'conversation.connect-request' + CREATE: 'conversation.create' + KNOCK: 'conversation.knock' + LOCATION: 'conversation.location' + MEMBER_JOIN: 'conversation.member-join' + MEMBER_LEAVE: 'conversation.member-leave' + MEMBER_UPDATE: 'conversation.member-update' + MESSAGE_ADD: 'conversation.message-add' + MESSAGE_DELETE: 'conversation.message-delete' + OTR_ASSET_ADD: 'conversation.otr-asset-add' + OTR_MESSAGE_ADD: 'conversation.otr-message-add' + RENAME: 'conversation.rename' + TYPING: 'conversation.typing' + VOICE_CHANNEL_ACTIVATE: 'conversation.voice-channel-activate' + VOICE_CHANNEL_DEACTIVATE: 'conversation.voice-channel-deactivate' + USER: + ACTIVATE: 'user.activate' + CONNECTION: 'user.connection' + UPDATE: 'user.update' + DELETE: 'user.delete' + CLIENT: + ADD: 'user.client-add' + REMOVE: 'user.client-remove' diff --git a/app/script/event/Client.coffee b/app/script/event/Client.coffee new file mode 100644 index 00000000000..ade047ff566 --- /dev/null +++ b/app/script/event/Client.coffee @@ -0,0 +1,24 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +z.event.Client = + CONVERSATION: + UNABLE_TO_DECRYPT: 'conversation.unable-to-decrypt' diff --git a/app/script/event/EventError.coffee b/app/script/event/EventError.coffee new file mode 100644 index 00000000000..1bccbf77af9 --- /dev/null +++ b/app/script/event/EventError.coffee @@ -0,0 +1,37 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +class z.event.EventError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @stack = (new Error()).stack + @type = type + + @:: = new Error() + @::constructor = @ + @::TYPE = + DATABASE_FAILURE: 'z.event.EventError::TYPE.DATABASE_FAILURE' + DATABASE_NOT_FOUND: 'z.event.EventError::TYPE.DATABASE_NOT_FOUND' + LAST_ID_NOT_SPECIFIED: 'z.event.EventError::TYPE.LAST_ID_NOT_SPECIFIED' + MISSING_CLIENT_ID: 'z.event.EventError::TYPE.MISSING_CLIENT_ID' + NO_NOTIFICATIONS: 'z.event.EventError::TYPE.NO_NOTIFICATIONS' + REQUEST_FAILURE: 'z.event.EventError::TYPE.REQUEST_FAILURE' diff --git a/app/script/event/EventRepository.coffee b/app/script/event/EventRepository.coffee new file mode 100644 index 00000000000..5b8ba80b8b1 --- /dev/null +++ b/app/script/event/EventRepository.coffee @@ -0,0 +1,393 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +# Event repository to handle all backend event channels. +class z.event.EventRepository + @::NOTIFICATION_SOURCE = + INJECTION: 'Injection' + SOCKET: 'WebSocket' + STREAM: 'Notification Stream' + + + ### + Construct a new Event Repository. + @param web_socket_service [z.event.WebSocketService] WebSocket service + @param notification_service [z.event.NotificationService] Service handling the notification stream + @param cryptography_repository [z.cryptography.CryptographyRepository] Repository for all cryptography interactions + @param user_repository [z.user.UserRepository] Repository for all user and connection interactions + ### + constructor: (@web_socket_service, @notification_service, @cryptography_repository, @user_repository) -> + @logger = new z.util.Logger 'z.event.EventRepository', z.config.LOGGER.OPTIONS + + @current_client = undefined + + @notifications_handled = 0 + @notifications_loaded = ko.observable false + @notifications_promises = [] + @notifications_total = 0 + @notifications_queue = ko.observableArray [] + @notifications_blocked = false + + @notifications_queue.subscribe (notifications) => + if notifications.length > 0 + return if @notifications_blocked + + notification = @notifications_queue()[0] + @notifications_blocked = true + @_handle_notification notification + .catch (error) => + @logger.log @logger.levels.WARN, 'We failed to handle a notification but will continue with queue', error + .then => + @notifications_blocked = false + @notifications_queue.shift() + @notifications_handled++ + + if @notifications_handled % 5 is 0 + replace = [@notifications_handled, @notifications_total] + amplify.publish z.event.WebApp.APP.UPDATE_INIT, z.string.init_events_progress, false, replace + + else if @notifications_loaded() and not @can_handle_web_socket() + @logger.log @logger.levels.INFO, "Done handling '#{@notifications_total}' notifications from the stream" + if @is_recovering() + @is_recovering false + else + amplify.publish z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, false + @find_ongoing_calls() + @can_handle_web_socket true + @notifications_loaded false + @notifications_promises[0] @last_notification_id() + + @web_socket_buffer = [] + @can_handle_web_socket = ko.observable false + @can_handle_web_socket.subscribe (was_handled) => + @_handle_buffered_notifications() if was_handled + + @last_notification_id = ko.observable undefined + @last_notification_id.subscribe (last_notification_id) => + @logger.log @logger.levels.INFO, "Last notification ID updated to '#{last_notification_id}'" + @notification_service.save_last_notification_id_to_db last_notification_id if last_notification_id + + @is_recovering = ko.observable false + @is_recovering.subscribe (is_recovering) => + if is_recovering + amplify.publish z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, true + @can_handle_web_socket false + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECOVERY + else + amplify.publish z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, false + @can_handle_web_socket true + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.CONNECTIVITY_RECOVERY + + amplify.subscribe z.event.WebApp.CONNECTION.RECONNECT, @reconnect + amplify.subscribe z.event.WebApp.CONNECTION.ONLINE, @recover_from_notification_stream + amplify.subscribe z.event.WebApp.EVENT.INJECT, @inject_event + + + ############################################################################### + # WebSocket handling + ############################################################################### + + # Initiate the WebSocket connection. + connect: => + if not @current_client().id + throw new z.event.EventError 'Missing client id', z.event.EventError::TYPE.MISSING_CLIENT_ID + + @web_socket_service.client_id = @current_client().id + @web_socket_service.connect (notification) => + if @can_handle_web_socket() + @notifications_queue.push notification + else + @_buffer_web_socket_notification notification + + ### + Close the WebSocket connection. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the disconnect + ### + disconnect: (trigger) => + @web_socket_service.reset trigger + + ### + Re-connect the WebSocket connection. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the disconnect + ### + reconnect: (trigger) => + @can_handle_web_socket false + @web_socket_service.reconnect trigger + + ### + Buffer an incoming notification. + @param notification [Object] Notification data + ### + _buffer_web_socket_notification: (notification) => + @web_socket_buffer.push notification + + # Handle buffered notifications. + _handle_buffered_notifications: => + @logger.log @logger.levels.INFO, "Received '#{@web_socket_buffer.length}' notifications via WebSocket while recovering from stream" + z.util.ko_array_push_all @notifications_queue, @web_socket_buffer + @web_socket_buffer.length = 0 + + + ############################################################################### + # Notification Stream handling + ############################################################################### + + ### + Get notifications for the current client from the stream. + @param notification_id [String] Event ID to start from + @return [Promise] Promise that resolves when all new notifications from the stream have been handled + ### + get_notifications: (last_notification_id, limit = 10000) -> + return new Promise (resolve, reject) => + _got_notifications = (response) => + if response.notifications.length > 0 + last_notification_id = response.notifications[response.notifications.length - 1].id + + notifications = (notification for notification in response.notifications) + @logger.log @logger.levels.INFO, "Added '#{notifications.length}' notifications to the queue" + z.util.ko_array_push_all @notifications_queue, notifications + + if @notifications_promises.length is 0 + @notifications_promises = [resolve, reject] + + @notifications_total += notifications.length + + if response.has_more + @get_notifications last_notification_id, 5000 + else + @notifications_loaded true + @logger.log @logger.levels.INFO, "Fetched '#{@notifications_total}' notifications from the backend" + amplify.publish z.event.WebApp.APP.UPDATE_INIT, z.string.init_events_expectation, true, [@notifications_total] + + else + error_message = "No notifications found since '#{last_notification_id}'" + @logger.log @logger.levels.INFO, error_message, response + reject new z.event.EventError error_message, z.event.EventError::TYPE.NO_NOTIFICATIONS + + @notification_service.get_notifications @current_client().id, last_notification_id, limit + .then (response) -> _got_notifications response + .catch (response) => + # When asking for notifications with a since set to a notification ID that does not belong to our client ID, + # we will get a 404 AND notifications + if response.notifications + _got_notifications response + else if error.code is z.service.BackendClientError::STATUS_CODE.NOT_FOUND + error_message = "No notifications found since '#{last_notification_id}'" + @logger.log @logger.levels.INFO, error_message, response + reject new z.event.EventError error_message, z.event.EventError::TYPE.NO_NOTIFICATIONS + else + error_message = "Failed to get notifications: #{error.message}" + @logger.log @logger.levels.ERROR, error_message, error + reject new z.event.EventError error_message, z.event.EventError::TYPE.REQUEST_FAILURE + + ### + Get the last notification ID for a given client. + @note This API endpoint is currently broken on the backend + @return [Promise] Promise that resolves with the last known notification ID matching a client + ### + get_last_notification_id: -> + return new Promise (resolve, reject) => + @notification_service.get_notifications_last @current_client?().id + .then (response) -> + resolve response.id + .catch reject + ### + Will retrieve missed notifications from the stream after a connectivity loss. + ### + recover_from_notification_stream: => + @is_recovering true + @update_from_notification_stream() + .then (number_of_notifications) => + @is_recovering false if number_of_notifications is 0 + @logger.log @logger.levels.INFO, "Retrieved '#{number_of_notifications}' notifications from stream after connectivity loss" + .catch (error) => + if error.type isnt z.event.EventError::TYPE.NO_NOTIFICATIONS + @logger.log @logger.levels.ERROR, "Failed to recover from notification stream: #{error.message}", error + @is_recovering false + # @todo What do we do in this case? + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + + ### + Fetch all missed events from the notification stream since the last ID stored in database. + @return [Promise] Promise that resolves with the total number of notifications + ### + update_from_notification_stream: => + return new Promise (resolve, reject) => + @notification_service.get_last_notification_id_from_db() + .then (last_notification_id) => + @last_notification_id last_notification_id + @notifications_total = 0 + return @get_notifications @last_notification_id(), 500 + .then (last_notification_id) => + if last_notification_id + @logger.log @logger.levels.INFO, "ID of last notification fetched from stream is '#{last_notification_id}'" + resolve @notifications_total + .catch (error) => + @can_handle_web_socket true + if error.type in [z.event.EventError::TYPE.NO_NOTIFICATIONS, z.event.EventError::TYPE.DATABASE_NOT_FOUND] + amplify.publish z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, false + @find_ongoing_calls() + @logger.log @logger.levels.INFO, 'No notifications found for this user', error + resolve 0 + else + @logger.log @logger.levels.ERROR, "Failed to handle notification stream: #{error.message}", error + reject error + + ### + Method to return an array of Conversation IDs which have a certain active conversation type. + + Example: + If the notifications for a conversation are for example "call.on", "call.off" and "call.on" then the call is active + because the last event which was seen was a "call.on". But if it would be "call.off" then the conversation would not + be marked as active and it's ID would not be returned. + + @param include_on [Array] List of event types to look for + @param exclude_on [Array] Remove activate state on these events + ### + get_conversation_ids_with_active_events: (include_on, exclude_on) => + return new Promise (resolve, reject) => + @cryptography_repository.storage_repository.load_events_by_types _.flatten [include_on, exclude_on] + .then (records) -> + raw_events = (record.raw for record in records) + + filtered_conversations = {} + + for event in raw_events + conversation_id = event.conversation + if event.type in include_on + filtered_conversations[conversation_id] = null + else if event.type in exclude_on + delete filtered_conversations[conversation_id] + resolve Object.keys filtered_conversations + .catch (error) => + @logger.log @logger.levels.ERROR, "Something failed: #{error?.message}", error + reject error + + ### + Check for conversations with ongoing calls. + @return [Promise] Promise that resolves when conversation that could contain a call have been identified + ### + find_ongoing_calls: => + @logger.log @logger.levels.INFO, 'Checking for ongoing calls' + @get_conversation_ids_with_active_events [z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE], [z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE] + .then (response) => + @logger.log @logger.levels.INFO, "Identified '#{response.length}' conversations that possibly have an ongoing call", response + amplify.publish z.event.WebApp.CALL.STATE.CHECK, conversation_id for conversation_id in response + .catch (error) => + @logger.log @logger.levels.ERROR, 'Could not check for active calls', error + + + ############################################################################### + # Notification/Event handling + ############################################################################### + + ### + Inject event into a conversation. + @note Don't add unable to decrypt to self conversation + @param event [Object] Event payload to be injected + ### + inject_event: (event) => + if event.conversation isnt @user_repository.self().id + @_handle_event event, @NOTIFICATION_SOURCE.INJECTION + + ### + Publish the given event. + @param event [Object] Mapped event + ### + _distribute_event: (event) -> + switch event.type.split('.')[0] + when 'call' + amplify.publish z.event.WebApp.CALL.EVENT_FROM_BACKEND, event + when 'conversation' + amplify.publish z.event.WebApp.CONVERSATION.EVENT_FROM_BACKEND, event + else + amplify.publish event.type, event + + if event.conversation + @logger.log @logger.levels.INFO, "Distributed '#{event.type}' event for conversation '#{event.conversation}'", event + else + @logger.log @logger.levels.INFO, "Distributed '#{event.type}' event", event + + ### + Handle a single event from the notification stream or WebSocket. + @param event [JSON] Backend event extracted from notification stream + @return [Promise] Promise that resolves with boolean whether the event was saved + ### + _handle_event: (event, source) -> + return new Promise (resolve, reject) => + sending_client = event.data?.sender + if sending_client + log_message = "Received encrypted event '#{event.type}' from client '#{sending_client}' of user '#{event.from}'" + else if event.from + log_message = "Received unencrypted event '#{event.type}' from user '#{event.from}'" + else + log_message = "Received call event '#{event.type}' in conversation '#{event.conversation}'" + @logger.log @logger.levels.INFO, log_message, {event_object: event, event_json: JSON.stringify event} + + if event.type in z.event.EventTypeHandling.IGNORE + @logger.log "Event ignored: '#{event.type}'", {event_object: event, event_json: JSON.stringify event} + return resolve true + else if event.type in z.event.EventTypeHandling.DECRYPT + promise = @cryptography_repository.decrypt_event(event).then (generic_message) => + @cryptography_repository.save_encrypted_event generic_message, event + else if event.type in z.event.EventTypeHandling.STORE + promise = @cryptography_repository.save_unencrypted_event event + else + promise = Promise.resolve {raw: event} + + promise.then (record) => + if record and (source is @NOTIFICATION_SOURCE.SOCKET or @is_recovering or record.raw.type.startsWith 'conversation') + @_distribute_event record.mapped or record.raw + resolve record + .catch (error) => + if error.type is z.cryptography.CryptographyError::TYPE.PREVIOUSLY_STORED + resolve true + else + @logger.log @logger.levels.ERROR, + "Failed to handle '#{event.type}' event '#{event.id or 'no ID'}' from '#{source}': '#{error.message}'", event + reject error + + ### + Handle all events from the payload of an incoming notification. + @param event [Object] Event data + @return [String] ID of the handled notification + ### + _handle_notification: (notification) => + return new Promise (resolve, reject) => + events = notification.payload + source = if @can_handle_web_socket() then @NOTIFICATION_SOURCE.SOCKET else @NOTIFICATION_SOURCE.STREAM + + @logger.log @logger.levels.INFO, + "Handling notification '#{notification.id}' from '#{source}' containing '#{events.length}' events", notification + + if events.length is 0 + @logger.log @logger.levels.WARN, 'Notification payload does not contain any events' + @last_notification_id notification.id + resolve @last_notification_id() + else + Promise.all (@_handle_event event, source for event in events) + .then => + @last_notification_id notification.id + resolve @last_notification_id() + .catch (error) => + @logger.log @logger.levels.ERROR, + "Failed to handle notification '#{notification.id}' from '#{source}': #{error.message}", error + reject error diff --git a/app/script/event/EventTypeHandling.coffee b/app/script/event/EventTypeHandling.coffee new file mode 100644 index 00000000000..53f7436cbe0 --- /dev/null +++ b/app/script/event/EventTypeHandling.coffee @@ -0,0 +1,43 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +z.event.EventTypeHandling = + DECRYPT: [ + z.event.Backend.CONVERSATION.OTR_ASSET_ADD + z.event.Backend.CONVERSATION.OTR_MESSAGE_ADD + ] + IGNORE: [ + z.event.Backend.CONVERSATION.CLIENT_MESSAGE_ADD + z.event.Backend.CONVERSATION.TYPING + ] + STORE: [ + z.event.Backend.CONVERSATION.ASSET_ADD + z.event.Backend.CONVERSATION.HOT_KNOCK + z.event.Backend.CONVERSATION.KNOCK + z.event.Backend.CONVERSATION.LOCATION + z.event.Backend.CONVERSATION.MEMBER_JOIN + z.event.Backend.CONVERSATION.MEMBER_LEAVE + z.event.Backend.CONVERSATION.MESSAGE_ADD + z.event.Backend.CONVERSATION.RENAME + z.event.Backend.CONVERSATION.VOICE_CHANNEL_ACTIVATE + z.event.Backend.CONVERSATION.VOICE_CHANNEL_DEACTIVATE + z.event.Client.CONVERSATION.UNABLE_TO_DECRYPT + ] diff --git a/app/script/event/NotificationService.coffee b/app/script/event/NotificationService.coffee new file mode 100644 index 00000000000..0e14e3f199c --- /dev/null +++ b/app/script/event/NotificationService.coffee @@ -0,0 +1,88 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +# Notification Service for all notification stream calls to the backend REST API. +class z.event.NotificationService + PRIMARY_KEY_LAST_NOTIFICATION: 'z.storage.StorageKey.NOTIFICATION.LAST_ID' + URL_NOTIFICATIONS: '/notifications' + URL_NOTIFICATIONS_LAST: '/notifications/last' + ### + Construct a new Notification Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client, @storage_service) -> + @logger = new z.util.Logger 'z.event.NotificationService', z.config.LOGGER.OPTIONS + + ### + Get notifications from the stream. + + @param size [Integer] Maximum number of notifications to return + @param client_id [String] Only return notifications targeted at the given client + @param since [String] Only return notifications more recent than this notification ID (like "7130304a-c839-11e5-8001-22000b0fe035") + @return [Promise] Promise that resolves with the notifications + ### + get_notifications: (client_id, notification_id, size) -> + @client.send_request + url: @client.create_url @URL_NOTIFICATIONS + type: 'GET' + data: + client: client_id + since: notification_id + size: size + + ### + Get the last notification for a given client. + @param client_id [String] Only return notifications targeted at the given client + ### + get_notifications_last: (client_id) -> + @client.send_request + url: @client.create_url @URL_NOTIFICATIONS_LAST + type: 'GET' + data: + client: client_id + + ### + Load last notifications id from storage. + @return [Promise] Promise that resolves with the stored last notification ID. + ### + get_last_notification_id_from_db: => + return new Promise (resolve, reject) => + @storage_service.load @storage_service.OBJECT_STORE_AMPLIFY, @PRIMARY_KEY_LAST_NOTIFICATION + .then (record) => + if record?.value + resolve record.value + else + error_message = 'Last notification ID not found in storage' + @logger.log @logger.levels.WARN, error_message + reject new z.event.EventError error_message, z.event.EventError::TYPE.DATABASE_NOT_FOUND + .catch (error) => + error_message = "Failed to get last notification ID from storage: #{error.message}" + @logger.log @logger.levels.ERROR, error_message, error + reject new z.event.EventError error_message, z.event.EventError::TYPE.DATABASE_FAILURE + + ### + Load last notifications id from storage. + @param notification_id [String] Notification ID to be stored + @return [Promise] Promise that will resolve with the stored record + ### + save_last_notification_id_to_db: (notification_id) => + payload = value: notification_id + return @storage_service.save @storage_service.OBJECT_STORE_AMPLIFY, @PRIMARY_KEY_LAST_NOTIFICATION, payload diff --git a/app/script/event/WebApp.coffee b/app/script/event/WebApp.coffee new file mode 100644 index 00000000000..605ba62c42d --- /dev/null +++ b/app/script/event/WebApp.coffee @@ -0,0 +1,187 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +# Enum of diffent webapp events. +z.event.WebApp = + ACTION: + SHOW: 'wire.webapp.action.show' + ANALYTICS: + EVENT: 'wire.webapp.analytics.event' + INIT: 'wire.webapp.analytics.init' + SESSION: + CLOSE: 'wire.webapp.analytics.session.close' + START: 'wire.webapp.analytics.session.start' + AUDIO: + PLAY: 'wire.webapp.audio.play' + PLAY_IN_LOOP: 'wire.webapp.audio.play-in-loop' + STOP: 'wire.webapp.audio.stop' + APP: + UPDATE_INIT: 'wire.webapp.app.update-init' + HIDE: 'wire.webapp.app.hide' + FADE_IN: 'wire.webapp.app.fade-in' + ARCHIVE: + SHOW: 'wire.webapp.archive.show' + CLOSE: 'wire.webapp.archive.close' + CALL: + EVENT_FROM_BACKEND: 'wire.webapp.call.event-from-backend' + STATE: + CHECK: 'wire.webapp.call.state.check' + DELETE: 'wire.webapp.call.state.delete' + IGNORE: 'wire.webapp.call.state.ignore' + JOIN: 'wire.webapp.call.state.join' + LEAVE: 'wire.webapp.call.state.leave' + REMOVE_PARTICIPANT: 'wire.webapp.call.state.remove-participant' + TOGGLE: 'wire.webapp.call.state.toggle' + TOGGLE_SCREEN: 'wire.webapp.call.state.toggle-screen' + MEDIA: + MUTE_AUDIO: 'wire.webapp.call.media.mute_audio' + ADD_STREAM: 'wire.webapp.call.media.add_stream' + SIGNALING: + DELETE_FLOW: 'wire.webapp.call.signaling.delete-flow' + POST_FLOWS: 'wire.webapp.call.signaling.post-flows' + SEND_ICE_CANDIDATE_INFO: 'wire.webapp.call.signaling.send-ice-candidate-info' + SEND_LOCAL_SDP_INFO: 'wire.webapp.call.signaling.send-local-sdp-info' + CLIENT: + DELETE: 'wire.webapp.client.delete' + CONNECT: + IMPORT_CONTACTS: 'wire.webapp.connect.import-contacts' + CONNECTION: + ACCESS_TOKEN: + RENEW: 'wire.webapp.connection.access-token.renew' + RENEWED: 'wire.webapp.connection.access-token.renewed' + RECONNECT: 'wire.webapp.connection.reconnect' + ONLINE: 'wire.webapp.connection.online' + CONVERSATION: + DEBUG: 'wire.webapp.conversation.debug' + EVENT_FROM_BACKEND: 'wire.webapp.conversation.event-from-backend' + LOADED_STATES: 'wire.webapp.conversation.loaded-states' + MAP_CONNECTION: 'wire.webapp.conversation.map-connection' + PEOPLE: + HIDE: 'wire.webapp.conversation.people.hide' + SHOW: 'wire.webapp.conversation.show' + STORE: 'wire.webapp.conversation.store' + SWITCH: 'wire.webapp.conversation.switch' + DETAIL_VIEW: + SHOW: 'wire.webapp.conversation.detail-view.show' + UNREAD: 'wire.webapp.conversation.unread' + ASSET: + CANCEL: 'wire.webapp.conversation.asset.cancel' + MESSAGE: + DELETE: 'wire.webapp.conversation.message.delete' + IMAGE: + SEND: 'wire.webapp.conversation.image.send' + CONVERSATION_LIST: + SHOW: 'wire.webapp.conversation-list.show' + ARCHIVE: + HIDE: 'wire.webapp.conversation-list.archive.hide' + DEBUG: + UPDATE_LAST_CALL_STATUS: 'wire.webapp.debug.update-last-call-status' + EXTENSIONS: + SHOW: 'wire.webapp.extionsions.show' + GIPHY: + SHOW: 'wire.webapp.extionsions.giphy.show' + SEND: 'wire.webapp.extionsions.giphy.send' + EVENT: + INJECT: 'wire.webapp.event.inject' + NOTIFICATION_HANDLING_STATE: 'wire.webapp.event.notification_handling' + LIST: + BLUR: 'wire.webapp.list.blur' + SCROLL: 'wire.webapp.list.scroll' + FULLSCREEN_ANIM_DISABLED: 'wire.webapp.list.anim-disabled' + LOADED: 'wire.webapp.loaded' + PEOPLE: + HIDE: 'wire.webapp.participant-et.hide' + SHOW: 'wire.webapp.participant-et.show' + TOGGLE: 'wire.webapp.participants.toggle' + PENDING: + SHOW: 'wire.webapp.pending.show' + LEFT: + HIDE: 'wire.webapp.left.hide' + FADE_IN: 'wire.webapp.left.fade-in' + LOGOUT: + ASK_TO_CLEAR_DATA: 'wire.webapp.logout.ask-to-clear-data' + WELCOME: + SHOW: 'wire.webapp.profile.welcome.show' + UNSPLASH_LOADED: 'wire.webapp.profile.welcome.unsplash-loaded' + PROFILE: + SHOW: 'wire.webapp.profile.show' + HIDE: 'wire.webapp.profile.hide' + FADE_IN: 'wire.webapp.profile.fade-in' + SETTINGS: + SHOW: 'wire.webapp.profile.settings.show' + UPLOAD_PICTURE: 'wire.webapp.profile.upload-picture' + PROPERTIES: + CHANGE: + APP_BANNER: 'wire.webapp.properties.change.app-banner' + DEBUG: 'wire.webapp.properties.change.debug' + UPDATE: + GOOGLE: 'wire.webapp.properties.update.google' + OSX_CONTACTS: 'wire.webapp.properties.update.google' + CALL_MUTE: 'wire.webapp.properties.update.call-mute' + SEND_DATA: 'wire.webapp.properties.update.send-data' + SOUND_ALERTS: 'wire.webapp.properties.update.sound-alerts' + HAS_CREATED_CONVERSATION: 'wire.webapp.properties.update.has-created-conversation' + UPDATED: 'wire.webapp.properties.updated' + SEARCH: + HIDE: 'wire.webapp.search.hide' + ONBOARDING: 'wire.webapp.search.onboarding' + SHOW: 'wire.webapp.people-picker.show' + BADGE: + HIDE: 'wire.webapp.search.badge.hide' + SHOW: 'wire.webapp.search.badge.show' + SIGN_OUT: 'wire.webapp.logout' + SYSTEM_NOTIFICATION: + CLICK: 'wire.webapp.system-notification.click' + NOTIFY: 'wire.webapp.system-notification.notify' + REMOVE_READ: 'wire.webapp.system.notification.remove_read' + REQUEST_PERMISSION: 'wire.webapp.system-notification.request_permission' + SHOW: 'wire.webapp.system-notification.show' + TELEMETRY: + BACKEND_REQUESTS: 'wire.webapp.telemetry.backend_requests' + USER: + UNBLOCKED: 'wire.webapp.user.unblocked' + EVENT_FROM_BACKEND: 'wire.webapp.user.event-from-backend' + WARNINGS: + SHOW: 'wire.webapp.warning.show' + DISMISS: 'wire.webapp.warning.dismiss' + MODAL: 'wire.webapp.warning.modal' + WINDOW: + RESIZE: + HEIGHT: 'wire.webapp.window.resize.height' + WIDTH: 'wire.webapp.window.resize.width' + SELF: + CLIENT_ADD: 'wire.webapp.self.client-add' + CLIENT_REMOVE: 'wire.webapp.self.client-remove' + SHORTCUT: + ADD_PEOPLE: 'wire.webapp.shortcut.add-people' + ARCHIVE: 'wire.webapp.shortcut.archive' + CALL_IGNORE: 'wire.webapp.shortcut.call-ignore' + CALL_MUTE: 'wire.webapp.shortcut.call-mute' + DEBUG: 'wire.webapp.shortcut.debug' + NEXT: 'wire.webapp.shortcut.next' + PEOPLE: 'wire.webapp.shortcut.people' + PICTURE: 'wire.webapp.shortcut.picture' + PING: 'wire.webapp.shortcut.ping' + PREV: 'wire.webapp.shortcut.prev' + SILENCE: 'wire.webapp.shortcut.silence' + START: 'wire.webapp.shortcut.start' + STORAGE: + SAVE_ENTITY: 'wire.webapp.storage.save-entity' diff --git a/app/script/event/WebSocketService.coffee b/app/script/event/WebSocketService.coffee new file mode 100644 index 00000000000..66f2f7c283e --- /dev/null +++ b/app/script/event/WebSocketService.coffee @@ -0,0 +1,186 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.event ?= {} + +RECONNECT_INTERVAL = 15000 + +PING_INTERVAL = 30000 +PING_INTERVAL_THRESHOLD = 2000 + +# WebSocket Service to manage the WebSocket connection to the backend. +class z.event.WebSocketService + @::CHANGE_TRIGGER = + CLEANUP: 'z.event.WebSocketService::CHANGE_TRIGGER.CLEANUP' + CLOSE: 'z.event.WebSocketService::CHANGE_TRIGGER.CLOSE' + ERROR: 'z.event.WebSocketService::CHANGE_TRIGGER.ERROR' + LOGOUT: 'z.event.WebSocketService::CHANGE_TRIGGER.LOGOUT' + OFFLINE: 'z.event.WebSocketService::CHANGE_TRIGGER.OFFLINE' + ONLINE: 'z.event.WebSocketService::CHANGE_TRIGGER.ONLINE' + PAGE_NAVIGATION: 'z.event.WebSocketService::CHANGE_TRIGGER.PAGE_NAVIGATION' + PING_INTERVAL: 'z.event.WebSocketService::CHANGE_TRIGGER.PING_INTERVAL' + READY_STATE: 'z.event.WebSocketService::CHANGE_TRIGGER.READY_STATE' + WARNING_BAR: 'z.event.WebSocketService::CHANGE_TRIGGER.WARNING_BAR' + + + ### + Construct a new WebSocket Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.event.WebSocketService', z.config.LOGGER.OPTIONS + + @client_id = undefined + @connection_url = '' + @socket = undefined + + @on_notification = undefined + + @ping_interval_id = undefined + @last_ping_time = undefined + + @reconnect_timeout_id = undefined + @reconnect_count = 0 + + ### + Establish the WebSocket connection. + @param on_notification [Function] Function to be called on incoming notifications + @return [Promise] Promise that resolves once the WebSocket connects + ### + connect: (on_notification) => + return new Promise (resolve, reject) => + @on_notification = on_notification + @connection_url = "#{@client.web_socket_url}/await?access_token=#{@client.access_token}" + @connection_url = z.util.append_url_parameter @connection_url, "client=#{@client_id}" if @client_id + + @reset z.event.WebSocketService::CHANGE_TRIGGER.CLEANUP if typeof @socket is 'object' + + @socket = new WebSocket @connection_url + @socket.binaryType = 'blob' + + # http://stackoverflow.com/a/27828483/451634 + delete @socket.URL + + @socket.onopen = => + @logger.log @logger.levels.INFO, "Connected WebSocket to: #{@client.web_socket_url}/await" + @ping_interval_id = window.setInterval @send_ping, PING_INTERVAL + resolve() + + @socket.onerror = (event) => + @logger.log @logger.levels.ERROR, 'WebSocket connection error.', event + @reset z.event.WebSocketService::CHANGE_TRIGGER.ERROR, true + + @socket.onclose = (event) => + @logger.log @logger.levels.WARN, 'Closed WebSocket connection', event + @reset z.event.WebSocketService::CHANGE_TRIGGER.CLOSE, true + + @socket.onmessage = (event) -> + if event.data instanceof Blob + blob_reader = new FileReader() + blob_reader.onload = -> on_notification JSON.parse blob_reader.result + blob_reader.readAsText event.data + + ### + Reconnect WebSocket after access token has been refreshed. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the reconnect + ### + pending_reconnect: (trigger) => + amplify.unsubscribeAll z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEWED + @logger.log @logger.levels.INFO, "Executing pending WebSocket reconnect triggered by '#{trigger}' after access token refresh" + @reconnect trigger + + ### + Try to re-establish the WebSocket connection. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the reconnect + ### + reconnect: (trigger) => + if not z.storage.get_value z.storage.StorageKey.AUTH.ACCESS_TOKEN.EXPIRATION + @logger.log @logger.levels.INFO, 'Access token has to be refreshed before reconnecting the WebSocket' + amplify.subscribe z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEWED, => @pending_reconnect trigger + return amplify.publish z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEW + + @reconnect_count++ + reconnect = => + @logger.log @logger.levels.INFO, "Trying to re-establish WebSocket connection. Try ##{@reconnect_count}" + @connect @on_notification + .then => + @reconnect_count = 0 + @logger.log @logger.levels.INFO, "Reconnected to WebSocket triggered by '#{trigger}'" + @on_socket_reconnected() + + if @reconnect_count is 1 + reconnect() + else + @reconnect_timeout_id = setTimeout -> + reconnect() + , RECONNECT_INTERVAL + + ### + Reset the WebSocket connection. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the reset + @param reconnect [Boolean] Re-establish the WebSocket connection + ### + reset: (trigger, reconnect = false) => + if @socket?.onclose + @logger.log @logger.levels.INFO, "WebSocket reset triggered by '#{trigger}'" + @socket.onerror = undefined + @socket.onclose = undefined + @socket.close() + window.clearInterval @ping_interval_id + window.clearTimeout @reconnect_timeout_id + if reconnect + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + @reconnect trigger + + # Send a WebSocket ping. + send_ping: => + if @socket.readyState is 1 + current_time = Date.now() + @last_ping_time ?= current_time + ping_interval_diff = @last_ping_time - current_time + + if ping_interval_diff > PING_INTERVAL + PING_INTERVAL_THRESHOLD + @logger.log @logger.levels.WARN, 'Ping interval check failed', event + @reconnect z.event.WebSocketService::CHANGE_TRIGGER.PING_INTERVAL + else + @logger.log @logger.levels.INFO, 'Sending ping to WebSocket' + @socket.send 'Wire is so much nicer with internet!' + else + @logger.log @logger.levels.WARN, "WebSocket connection is closed. Current ready state: #{@socket.readyState}" + @reconnect z.event.WebSocketService::CHANGE_TRIGGER.READY_STATE + + ### + Behavior when WebSocket connection is re-established after a connection drop. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the reconnect + ### + on_socket_reconnected: => + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + @logger.log @logger.levels.WARN, 'Re-established WebSocket connection. Recovering from Notification Stream...' + amplify.publish z.event.WebApp.CONNECTION.ONLINE + + ### + Behavior when WebSocket connection is closed. + @param trigger [z.event.WebSocketService::CHANGE_TRIGGER] Trigger of the connection close + ### + on_socket_closed: (trigger) -> + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + error = new Error "WebSocket connection lost: #{trigger}" + custom_data = + network_status: navigator.onLine + Raygun.send error, custom_data diff --git a/app/script/extension/GiphyContentSizes.coffee b/app/script/extension/GiphyContentSizes.coffee new file mode 100644 index 00000000000..fa29879b26d --- /dev/null +++ b/app/script/extension/GiphyContentSizes.coffee @@ -0,0 +1,38 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.extension ?= {} + +# Enum of different Giphy content sizes. +z.extension.GiphyContentSizes = + FIXED_HEIGHT: 'fixed_height' + FIXED_HEIGHT_STILL: 'fixed_height_still' + FIXED_HEIGHT_DOWNSAMPLED: 'fixed_height_downsampled' + FIXED_WIDTH: 'fixed_width' + FIXED_WIDTH_STILL: 'fixed_width_still' + FIXED_WIDTH_DOWNSAMPLED: 'fixed_width_downsampled' + FIXED_HEIGHT_SMALL: 'fixed_height_small' + FIXED_HEIGHT_SMALL_STILL: 'fixed_height_small_still' + FIXED_WIDTH_SMALL: 'fixed_width_small' + FIXED_WIDTH_SMALL_STILL: 'fixed_width_small_still' + DOWNSIZED: 'downsized' + DOWNSIZED_STILL: 'downsized_still' + DOWNSIZED_LARGE: 'downsized_large' + ORIGINAL: 'original' + ORIGINAL_STILL: 'original_still' diff --git a/app/script/extension/GiphyRepository.coffee b/app/script/extension/GiphyRepository.coffee new file mode 100644 index 00000000000..7a830be70c9 --- /dev/null +++ b/app/script/extension/GiphyRepository.coffee @@ -0,0 +1,143 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.extension ?= {} + +# Giphy repository for all interactions with the giphy service. +class z.extension.GiphyRepository + ### + Construct a new Giphy Repository. + + @param giphy_service [z.extension.GiphyService] Giphy REST API implementation + ### + constructor: (@giphy_service) -> + @logger = new z.util.Logger 'z.extension.GiphyRepository', z.config.LOGGER.OPTIONS + @gif_query_cache = {} + + ### + Get random GIF for a word or phrase. + + @param options [Object] + @option options [String] tag search query term or phrase + @option options [Number] retry (optional) How many retries to get the correct size. (default 3) + @option options [Number] max_size (optional) Maximum gif size in bytes (default 3MB) + ### + get_random_gif: (options) -> + return new Promise (resolve, reject) => + options = $.extend + retry: 3 + max_size: 3 * 1024 * 1024 + , options + + _get_random_gif = (retries = 0) => + if options.retry is retries + reject new Error "Unable to fetch a proper gif within #{options.retry} retries" + + @giphy_service.get_random tag: options.tag + .then (response) => + @giphy_service.get_by_id ids: response.data.id + .then (response) => + images = response.data.images + static_gif = images[z.extension.GiphyContentSizes.FIXED_WIDTH_STILL] + animation_gif = images[z.extension.GiphyContentSizes.DOWNSIZED] + + if animation_gif.size > options.max_size + @logger.log "Gif size (#{animation_gif.size}) is over maximum size (#{animation_gif.size})" + _get_random_gif retries + 1 + else + resolve + url: response.data.url + static: static_gif.url + animated: animation_gif.url + .catch (error) -> + reject error + + _get_random_gif() + + ### + Get random GIFs for a word or phrase. + + @param options [Object] + @option options [String] query search query term or phrase + @option options [Number] number amount of GIFs to receive + @option options [Number] max_size (optional) Maximum gif size in bytes (default 3MB) + @option options [Boolean] random (optional) will return an randomized result (default true) + @option options [String] sorting (optional) specify sorting ('relevant' or 'recent' default 'relevant') + ### + get_gifs: (options) => + return new Promise (resolve, reject) => + offset = 0 + result = [] + + options = $.extend + number: 6 + max_size: 3 * 1024 * 1024 + random: true + sorting: 'relevant' + , options + + if not options.query + error = new Error 'No query specified' + @logger.log @logger.levels.ERROR, error.message, error + reject error + + if options.random + options.sorting = z.util.ArrayUtil.random_element ['recent', 'relevant'] + + total = @gif_query_cache[options.query] + if total? + if options.number >= total + offset = 0 + else + range = total - options.number + offset = Math.floor Math.random() * range + + @giphy_service.get_search + query: options.query + limit: 100 + sorting: options.sorting + offset: offset + .then (response) => + + gifs = response.data + if options.random + gifs = gifs.sort -> .5 - Math.random() + + @gif_query_cache[options.query] = response.pagination.total_count + + for n in [0...options.number] + break if n is gifs.length + + gif = gifs[n] + images = gif.images + static_gif = images[z.extension.GiphyContentSizes.FIXED_WIDTH_STILL] + animation_gif = images[z.extension.GiphyContentSizes.DOWNSIZED] + + if animation_gif.size > options.max_size + continue + else + result.push + url: gif.url + static: static_gif.url + animated: animation_gif.url + + resolve result + .catch (error) => + @logger.log "Unable to fetch gif for query: #{options.query}", error + reject error diff --git a/app/script/extension/GiphyService.coffee b/app/script/extension/GiphyService.coffee new file mode 100644 index 00000000000..15e615687d6 --- /dev/null +++ b/app/script/extension/GiphyService.coffee @@ -0,0 +1,90 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.extension ?= {} + +# Giphy Service for all giphy calls to the backend REST API. +class z.extension.GiphyService + ### + Construct a new Giphy Service. + + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @GIPHY_ENDPOINT_BASE = '/giphy/v1/gifs' + + ### + Get GIFs for IDs. + + @param [Object] + @option options [String|Array] ids A single id or comma separated list of IDs to fetch GIF size data. + @option options [Function] callback (optional) Function to be called on server return + ### + get_by_id: (options) => + ids = if _.isArray options.ids then options.ids else [options.ids] + url = "#{@GIPHY_ENDPOINT_BASE}/#{ids.join ','}" + + @client.send_json + type: 'GET' + url: @client.create_url url + callback: options.callback + + ### + Search all Giphy GIFs for a word or phrase. + + @param [Object] + @option options [String] tag The GIF tag to limit randomness by + @option options [Function] callback (optional) Function to be called on server return + ### + get_random: (options) => + url = "#{@GIPHY_ENDPOINT_BASE}/random?tag=#{encodeURIComponent options.tag}" + + @client.send_json + type: 'GET' + url: @client.create_url url + callback: options.callback + + ### + Search GIFs for a word or phrase. + + @param options [Object] + @option options [String] query search query term or phrase + @option options [Number] limit (optional) Number of results to return (maximum 100, default 25) + @option options [Number] offset (optional) Results offset (defaults 0) + @option options [String] sorting (optional) Relevant or recent + @option options [Function] callback (optional) Function to be called on server return + ### + get_search: (options) -> + + options = $.extend + limit: 25 + offset: 0 + sorting: 'relevant' + , options + + url = "#{@GIPHY_ENDPOINT_BASE}/search" + + "?q=#{encodeURIComponent options.query}" + + "&offset=#{options.offset}" + + "&limit=#{options.limit}" + + "&sort=#{options.sorting}" + + @client.send_json + type: 'GET' + url: @client.create_url url + callback: options.callback diff --git a/app/script/links/LinkPreviewError.coffee b/app/script/links/LinkPreviewError.coffee new file mode 100644 index 00000000000..3fcd7112325 --- /dev/null +++ b/app/script/links/LinkPreviewError.coffee @@ -0,0 +1,33 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.links ?= {} + +class z.links.LinkPreviewError + constructor: (type) -> + @name = @constructor.name + @stack = (new Error()).stack + @type = type + + @:: = new Error() + @::constructor = @ + @::TYPE = + NOT_SUPPORTED: 'z.links.LinkPreviewError::TYPE.NOT_SUPPORTED' + UNSUPPORTED_TYPE: 'z.links.LinkPreviewError::TYPE.UNSUPPORTED_TYPE' + NO_DATA_AVAILABLE: 'z.links.LinkPreviewError::TYPE.NO_DATA_AVAILABLE' diff --git a/app/script/links/LinkPreviewHelpers.coffee b/app/script/links/LinkPreviewHelpers.coffee new file mode 100644 index 00000000000..cf1e12632c9 --- /dev/null +++ b/app/script/links/LinkPreviewHelpers.coffee @@ -0,0 +1,45 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.links ?= {} + +z.links.LinkPreviewHelpers = + + ### + Check if the text contains only one link + + @param text [String] + ### + contains_only_link: (text) -> + text = text.trim() + urls = twttr.txt.extractUrls text + return urls.length is 1 and urls[0] is text + + ### + Get first link and link offset for given text. + + @param text [String] + ### + get_first_link_with_offset: (text) -> + links = twttr.txt.extractUrls text + first_link = links[0] + + if first_link? + link_offset = text.indexOf first_link + return [first_link, link_offset] diff --git a/app/script/links/LinkPreviewProtoBuilder.coffee b/app/script/links/LinkPreviewProtoBuilder.coffee new file mode 100644 index 00000000000..fd178c44f26 --- /dev/null +++ b/app/script/links/LinkPreviewProtoBuilder.coffee @@ -0,0 +1,47 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.links ?= {} + +z.links.LinkPreviewProtoBuilder = do -> + + ### + Create link preview proto message + + @param data [Object] open graph data + @param url [String] link entered by the user + @param offset [Number] starting index of the link + + @returns [z.proto.LinkPreview] + ### + build_from_open_graph_data = (data, url, offset = 0) -> + preview = null + + return if _.isEmpty data + + switch + when not data.type? or data.type in ['article', 'website'] and data.title + preview = new z.proto.Article data.url or url, data.title, data.description + + if preview? + return new z.proto.LinkPreview url, offset, preview + + return { + build_from_open_graph_data: build_from_open_graph_data + } diff --git a/app/script/links/LinkPreviewRepository.coffee b/app/script/links/LinkPreviewRepository.coffee new file mode 100644 index 00000000000..a486bff3c08 --- /dev/null +++ b/app/script/links/LinkPreviewRepository.coffee @@ -0,0 +1,89 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.links ?= {} + +class z.links.LinkPreviewRepository + constructor: (@asset_service) -> + @logger = new z.util.Logger 'z.links.LinkPreviewRepository', z.config.LOGGER.OPTIONS + + ### + Create link preview for given link. This will upload associated image as asset and will + resolve with an z.proto.LinkPreview instance + + @param url [String] + @param offset [Number] starting index of the link + ### + get_link_preview: (url, offset = 0) -> + open_graph_data = null + + Promise.resolve() + .then => + if window.openGraph + return @_fetch_open_graph_data url + else + throw new z.links.LinkPreviewError z.links.LinkPreviewError::TYPE.NOT_SUPPORTED + .then (data) -> + open_graph_data = data + if open_graph_data + return z.links.LinkPreviewProtoBuilder.build_from_open_graph_data data, url, offset + else + throw new z.links.LinkPreviewError z.links.LinkPreviewError::TYPE.NO_DATA_AVAILABLE + .then (link_preview) -> + if link_preview? + return link_preview + else + throw new z.links.LinkPreviewError z.links.LinkPreviewError::TYPE.UNSUPPORTED_TYPE + .then (link_preview) => + return @_fetch_preview_image link_preview, open_graph_data.image + + ### + Fetch and upload open graph images + + ### + _fetch_preview_image: (link_preview, open_graph_image) -> + if link_preview.preview is 'article' and open_graph_image?.data + return @_upload_preview_image open_graph_image.data + .then (asset) -> + link_preview.article.set 'image', asset + return link_preview + else + return link_preview + + ### + Fetch open graph data + + @param url [String] + ### + _fetch_open_graph_data: (link) -> + return new Promise (resolve, reject) -> + openGraph link, (error, data) -> + if error then reject error else resolve data + + ### + Upload open graph image as asset + + @param data_uri [String] image data as base64 encoded data URI + ### + _upload_preview_image: (data_URI) -> + Promise.resolve() + .then -> + return z.util.base64_to_blob data_URI + .then (blob) => + @asset_service.upload_image_asset blob, public: true diff --git a/app/script/localization/Localizer.coffee b/app/script/localization/Localizer.coffee new file mode 100644 index 00000000000..1da23e07909 --- /dev/null +++ b/app/script/localization/Localizer.coffee @@ -0,0 +1,106 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.localization ?= {} + +# Localizer to replace strings. +class Localizer + # Construct a new Localizer. + constructor: -> + param = z.util.get_url_parameter z.auth.URLParameter.LOCALE + z.storage.set_value z.storage.StorageKey.LOCALIZATION.LOCALE, param if param + @locale = z.storage.get_value(z.storage.StorageKey.LOCALIZATION.LOCALE) or navigator.language.substr(0, 2) or 'en' + # Moment defaults to the language loaded last. Thus we need to set the fallback to English until we use all locales. + # @see http://momentjs.com/docs/#/i18n/changing-locale/ + moment.locale [@locale, 'en'] + $.extend z.string, z.string[@locale] if z.string[@locale] + + ### + Pulls the localized string from the resources and replaces placeholders. + + @note Takes the id of the string for look up from z.string is directly for simple use. Else pass it in as the id + parameter in conjunction with a single or multiple (it supports but does not require an array) replace rules that + consist of a placeholder and the content that it should be replace with. + + @param id [String] Localization string ID + @param replace [Object | Array] Placeholders that should be replaced + @option replace [String] placeholder Content to be replaced + @option replace [String] content replacing content + ### + get_text: (valueAccessor) -> + return if not valueAccessor? + args = [] + + if valueAccessor.id? + s = valueAccessor.id + if _.isArray valueAccessor.replace + args = valueAccessor.replace + else + args.push valueAccessor.replace + else + s = valueAccessor + + if args.length isnt 0 + for i in [0...args.length] + reg = new RegExp args[i].placeholder, 'gm' + s = s.replace reg, args[i].content + + return s + +z.localization.Localizer = new Localizer() + + +# Knockout binding to localize links. +ko.bindingHandlers.l10n_href = + update: (element, valueAccessor) -> + element.setAttribute 'href', z.localization.Localizer.get_text valueAccessor() + + +# Knockout binding to localize input values. +ko.bindingHandlers.l10n_input = + update: (element, valueAccessor) -> + element.setAttribute 'value', z.localization.Localizer.get_text valueAccessor() + + +# Knockout binding to localize input placeholders. +ko.bindingHandlers.l10n_placeholder = + update: (element, valueAccessor) -> + element.setAttribute 'placeholder', z.localization.Localizer.get_text valueAccessor() + + +# Knockout binding to localize element text. +ko.bindingHandlers.l10n_text = + update: (element, valueAccessor) -> + ko.utils.setTextContent element, z.localization.Localizer.get_text valueAccessor() + +# Knockout binding to localize element html content. +ko.bindingHandlers.l10n_html = + update: (element, valueAccessor) -> + ko.utils.setHtml element, z.localization.Localizer.get_text valueAccessor() + +# Knockout binding to localize element tooltips. +ko.bindingHandlers.l10n_tooltip = + update: (element, valueAccessor) -> + element.setAttribute 'title', z.localization.Localizer.get_text valueAccessor() + + +# Knockout binding to localize element tooltips. +ko.bindingHandlers.l10n_aria_label = + update: (element, valueAccessor) -> + element.setAttribute 'aria-label', z.localization.Localizer.get_text valueAccessor() diff --git a/app/script/localization/strings-de.coffee b/app/script/localization/strings-de.coffee new file mode 100755 index 00000000000..b7c42368bf8 --- /dev/null +++ b/app/script/localization/strings-de.coffee @@ -0,0 +1,566 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +#General terms +z.string.de.wire = 'Wire' +z.string.de.wire_osx = 'Wire für OS X' +z.string.de.wire_windows = 'Wire für Windows' +z.string.de.truncation = '…' +z.string.de.nonexistent_user = 'Gelöschte Person' +z.string.de.and = 'und' + +# About screen +z.string.de.about_copyright = '© Wire Swiss GmbH • Version' +z.string.de.about_legal = 'Datenschutz u. AGB' +z.string.de.about_privacy = 'Privatsphäre' +z.string.de.about_wire = 'wire.com' + +# Alert view when trying to set a profile image that's too small +z.string.de.alert_upload_file_format = 'Das Bild kann nicht verwendet werden. Bitte wähle eine PNG- oder JPEG-Datei.' +z.string.de.alert_upload_too_small = 'Das Bild kann nicht verwendet werden. Bitte wähle ein Bild mit mindestens 320 x 320 Pixeln.' +z.string.de.alert_upload_too_large = 'Das Bild ist zu groß. Du kannst Dateien bis zu %no MB hochladen.' +z.string.de.alert_gif_too_large = 'Das GIF ist zu groß. Die maximale Größe beträgt %no MB.' + +# Auth +# Authentication: ACCOUNT section +z.string.de.auth_account_country_code = 'Landesvorwahl' +z.string.de.auth_account_create = 'Erstellen' +z.string.de.auth_account_create_account = 'Konto erstellen' +z.string.de.auth_account_expiration = 'Du wurdest abgemeldet, da deine Session abgelaufen ist. Bitte logge dich erneut ein.' +z.string.de.auth_account_get_wire = 'Ein moderner und sicherer Messenger für Unterhaltungen, Anrufe, Bilder, Musik, Videos, GIFs und mehr.' +z.string.de.auth_account_password_forgot = 'Passwort vergessen' +z.string.de.auth_account_remember_me = 'Angemeldet bleiben' +z.string.de.auth_account_sign_in = 'Login' +z.string.de.auth_account_sign_in_email = 'E-Mail' +z.string.de.auth_account_sign_in_phone = 'Handy' +z.string.de.auth_account_terms_of_use = 'Nutzungsbedingungen' +z.string.de.auth_account_terms_of_use_detail = 'Ich akzeptiere die' + +# Authentication: VERIFY section +z.string.de.auth_verify_code_description = 'Gib den Code ein, den wir an \n%@number gesendet haben.' +z.string.de.auth_verify_code_resend = 'Keinen Code bekommen?' +z.string.de.auth_verify_code_resend_detail = 'Erneut senden' +z.string.de.auth_verify_code_resend_timer = 'Du kannst %expiration einen neuen Code anfordern.' +z.string.de.auth_verify_code_change_phone = 'Telefonnummer ändern' +z.string.de.auth_verify_email_button = 'Hinzufügen' +z.string.de.auth_verify_email_detail = 'Um Wire auf mehreren Geräten benutzen zu können, werden eine E-Mail-Adresse und Passwort benötigt.' +z.string.de.auth_verify_email_headline = 'Hallo, %name.' + +# Authentication: limit section +z.string.de.auth_limit_devices_headline = 'Geräte' +z.string.de.auth_limit_description = 'Entferne eines deiner anderen Geräte, um Wire hier zu nutzen.' +z.string.de.auth_limit_button_manage = 'Geräte verwalten' +z.string.de.auth_limit_button_sign_out = 'Abmelden' +z.string.de.auth_limit_devices_current = '(Aktuelles Gerät)' + +# Authentication: limit section +z.string.de.auth_history_headline = 'Du benutzt Wire zum ersten Mal auf diesem Gerät.' +z.string.de.auth_history_description = 'Aus Datenschutzgründen wird dein bisheriger Gesprächsverlauf nicht angezeigt.' +z.string.de.auth_history_button = 'Verstanden' + +# Authentication: POSTED section +z.string.de.auth_posted_change_email = 'E-Mail-Adresse ändern' +z.string.de.auth_posted_offline_detail = 'Überprüfe deine Internetverbindung und versuche es erneut.' +z.string.de.auth_posted_offline_headline = 'Wire ist online schöner.' +z.string.de.auth_posted_pending_detail = 'Schaue in deinen Posteingang oder sende die E-Mail zur Aktivierung erneut.' +z.string.de.auth_posted_pending_headline = 'Konto bereits erstellt' +z.string.de.auth_posted_resend = 'Erneut an %email senden' +z.string.de.auth_posted_resend_action = 'E-Mail nicht erhalten?' +z.string.de.auth_posted_resend_detail = 'Schaue in deinen Posteingang und folge den Anweisungen.' +z.string.de.auth_posted_resend_headline = 'Du hast Post.' +z.string.de.auth_posted_retry = 'E-Mail erneut an %email senden' +z.string.de.auth_posted_retry_action = 'Erneut versuchen?' +z.string.de.auth_posted_retry_detail = 'Versuche es erneut.' +z.string.de.auth_posted_retry_headline = 'Ein Fehler ist aufgetreten' +z.string.de.auth_posted_verify_later = 'Später verifizieren' + +#Authentication: Misc +z.string.de.auth_placeholder_email = 'E-Mail-Adresse' +z.string.de.auth_placeholder_name = 'Name' +z.string.de.auth_placeholder_password_put = 'Passwort' +z.string.de.auth_placeholder_password_set = 'Passwort (min. acht Zeichen)' +z.string.de.auth_placeholder_phone = 'Telefonnummer' +z.string.de.auth_hello = 'Hallo, %name.' + +# Authentication: Validation errors +z.string.de.auth_error_code = 'Ungültiger Verifizierungs-Code' +z.string.de.auth_error_country_code_invalid = 'Ungültige Landesvorwahl' +z.string.de.auth_error_email_exists = 'E-Mail-Adresse bereits vergeben' +z.string.de.auth_error_email_forbidden = 'Es tut uns leid. Diese E-Mail-Adresse ist verboten.' +z.string.de.auth_error_email_malformed = 'Bitte gib eine gültige E-Mail-Adresse ein.' +z.string.de.auth_error_email_missing = 'Bitte gib eine E-Mail-Adresse ein.' +z.string.de.auth_error_misc = 'Probleme mit der Verbindung. Versuche es erneut.' +z.string.de.auth_error_name_long = 'Gib einen kürzeren Namen ein' +z.string.de.auth_error_name_short = 'Gib deinen Namen mit mindestens zwei Zeichen ein' +z.string.de.auth_error_offline = 'Keine Internetverbindung' +z.string.de.auth_error_password_long = 'Das eingegebene Passwort ist zu lang' +z.string.de.auth_error_password_short = 'Wähle ein Passwort mit mindestens acht Zeichen.' +z.string.de.auth_error_password_wrong = 'Falsches Passwort. Bitte versuche es erneut.' +z.string.de.auth_error_phone_number_invalid = 'Ungültige Telefonnummer' +z.string.de.auth_error_phone_number_unknown = 'Unbekannte Telefonnummer' +z.string.de.auth_error_sign_in = 'Überprüfe deine Eingaben und versuche es erneut.' +z.string.de.auth_error_terms_of_use = 'Bitte akzeptiere die Nutzungsbedingungen.' + +# Call stuff +z.string.de.call_state_outgoing = 'Klingelt…' +z.string.de.call_state_connecting = 'Verbinde…' +z.string.de.call_state_incoming = 'Klingelt…' +z.string.de.call_decline = 'Ablehnen' +z.string.de.call_accept = 'Annehmen' +z.string.de.call_join = 'Beitreten' +z.string.de.call_choose_shared_screen = 'Wähle eine Bildschirm aus' + +# Calling Bar +z.string.de.call_banner_connecting = 'Verbinde…' +z.string.de.call_banner_in = 'in' +z.string.de.call_banner_incoming_1to1 = '%s.first_name ruft an' +z.string.de.call_banner_incoming_video_1to1 = '%s.first_name ruft mit Video an' +z.string.de.call_banner_incoming_group = '%s.first_name ruft %@.group an' +z.string.de.call_banner_join = 'Anruf annehmen' +z.string.de.call_banner_ongoing = 'Aktiver Anruf' +z.string.de.call_banner_outgoing = 'Rufe %@.first_name an' + +# Warnings +z.string.de.modal_button_cancel = 'Abbrechen' +z.string.de.modal_button_ok = 'Ok' + +# Block a user +z.string.de.modal_block_conversation_headline = '%@.name blockieren?' +z.string.de.modal_block_conversation_message = '%@.name wird dich auf Wire nicht finden können.' +z.string.de.modal_block_conversation_button = 'Blockieren' +# Cannot create the call because there is nobody to call (conversation_empty) +z.string.de.modal_call_conversation_empty_headline = 'Niemand um anzurufen' +z.string.de.modal_call_conversation_empty_message = 'Es ist niemand mehr in der Unterhaltung.' +# Cannot create the call because there are too many participants (conversation_full) +z.string.de.modal_call_conversation_full_headline = 'Zu viele Teilnehmer' +z.string.de.modal_call_conversation_full_message = 'Anrufe sind nur in Unterhaltungen mit bis zu %no Teilnehmern möglich.' +# Cannot video call in group conversations +z.string.de.modal_call_no_video_in_group_headline = 'Keine Videoanrufe in Gruppen' +z.string.de.modal_call_no_video_in_group_message = 'Videoanrufe sind in Gruppen nicht verfügbar.' +# Second incoming call +z.string.de.modal_call_second_incoming_headline = 'Anruf annehmen?' +z.string.de.modal_call_second_incoming_message = 'Dein aktueller Anruf wird beendet.' +z.string.de.modal_call_second_incoming_action = 'Annehmen' +# Second outgoing call +z.string.de.modal_call_second_outgoing_headline = 'Aktuellen Anruf beenden?' +z.string.de.modal_call_second_outgoing_message = 'Nur ein zeitgleicher Anruf möglich.' +z.string.de.modal_call_second_outgoing_action = 'Beenden' +# Cannot join the call because there are too many participants (voice_channel_full) +z.string.de.modal_call_voice_channel_full_headline = 'Volles Haus' +z.string.de.modal_call_voice_channel_full_message = 'Die maximale Teilnehmeranzahl beträgt %no Personen.' +# Clear a conversation +z.string.de.modal_clear_conversation_headline = 'Unterhaltungsverlauf "%@.name" löschen?' +z.string.de.modal_clear_conversation_message = 'Der Verlauf wird geleert und die Unterhaltung aus der Liste entfernt.' +z.string.de.modal_clear_conversation_option = 'Unterhaltung auch verlassen' +z.string.de.modal_clear_conversation_button = 'Löschen' +# Connected device +z.string.de.modal_connected_device_headline = 'Dein Benutzerkonto wurde verwendet:' +z.string.de.modal_connected_device_from = 'Mit:' +z.string.de.modal_connected_device_message = 'Falls du dieses Gerät nicht hinzugefügt hast, entferne es und setze dein Passwort zurück.' +z.string.de.modal_connected_device_manage_devices = 'Geräte verwalten' +# Delete message +z.string.de.modal_delete_button = 'Löschen' +z.string.de.modal_delete_headline = 'Nachricht löschen' +z.string.de.modal_delete_message = 'Die Nachricht wird nur auf deiner Seite der Unterhaltung gelöscht. Dies kann nicht rückgängig gemacht werden.' +# Too long message +z.string.de.modal_too_long_headline = 'Nachricht zu lang' +z.string.de.modal_too_long_message = 'Du kannst Nachrichten mit bis zu %no Zeichen senden.' +# Leave a conversation +z.string.de.modal_leave_conversation_headline = 'Unterhaltung "%@.name" verlassen?' +z.string.de.modal_leave_conversation_message = 'Die Personen werden benachrichtigt und die Unterhaltung aus deiner Liste entfernt.' +z.string.de.modal_leave_conversation_button = 'Verlassen' +# Logout +z.string.de.modal_logout_headline = 'Daten löschen?' +z.string.de.modal_logout_message = 'Dies entfernt alle deine persönlichen Informationen und Unterhaltungen von diesem Gerät.' +z.string.de.modal_logout_button = 'Abmelden' +# New device +z.string.de.modal_new_device_headline = '"%@.name" hat begonnen ein neues Gerät zu nutzen' +z.string.de.modal_new_device_message = 'Möchtest du die Nachrichten noch senden?' +z.string.de.modal_new_device_show_device = 'Gerät anzeigen' +z.string.de.modal_new_device_send_anyway = 'Dennoch senden' +# Session Reset +z.string.de.modal_session_reset_headline = 'Die Session wurde zurückgesetzt' +z.string.de.modal_session_reset_message_1 = 'Wenn das Problem weiterhin besteht,' +z.string.de.modal_session_reset_message_link = 'kontaktiere' +z.string.de.modal_session_reset_message_2 = 'uns.' +# Too many members in conversation +z.string.de.modal_too_many_members_headline = 'Volles Haus' +z.string.de.modal_too_many_members_message = 'An einer Unterhaltung für eine Gruppe können bis zu %max Personen teilnehmen. Hier ist noch Platz für %no Personen.' +# Whitelist screensharing +z.string.de.modal_whitelist_screensharing_headline = 'Whitelist us for screen sharing' +z.string.de.modal_whitelist_screensharing_message_1 = 'Until Firefox adds us to the list, manually add the current domain on "about:config" to "media.getusermedia.screensharing.allowed_domains". Follow the' +z.string.de.modal_whitelist_screensharing_message_link = 'FAQ' +z.string.de.modal_whitelist_screensharing_message_2 = 'that guides you through the process.' +# Parallel uploads +z.string.de.modal_uploads_parallel = 'Du kannst bis zu %no Dateien auf einmal senden.' + +# Connection requests +z.string.de.connection_request_connect = 'Kontakt hinzufügen' +z.string.de.connection_request_ignore = 'Ignorieren' +z.string.de.connection_request_message = 'Hallo %@.first_name,\nbitte füge mich als Kontakt hinzu.\n%s.first_name' + +# Conversation +z.string.de.conversation_you_nominative = 'du' +z.string.de.conversation_you_dative = 'dir' +z.string.de.conversation_you_accusative = 'dich' + +z.string.de.conversation_connection_accepted = 'Hinzugefügt' +z.string.de.conversation_connection_cancel_request = 'Kontaktanfrage abbrechen' +z.string.de.conversation_connection_blocked = 'Blockiert' +z.string.de.conversation_connection_pending = 'Ausstehend' +z.string.de.conversation_create = ' hat eine Unterhaltung mit %@names begonnen' +z.string.de.conversation_create_you = ' hast eine Unterhaltung mit %@names begonnen' +z.string.de.conversation_device_started_using = ' hat begonnen' +z.string.de.conversation_device_started_using_you = ' hast begonnen' +z.string.de.conversation_device_unverified = ' hast die Überprüfung von einem Gerät widerrufen für' +z.string.de.conversation_device_your_devices = ' deine Geräte' +z.string.de.conversation_device_user_devices = ' %@names Geräte' +z.string.de.conversation_device_a_new_device = ' ein neues Gerät' +z.string.de.conversation_device_this_device = ' dieses Gerät' +z.string.de.conversation_just_now = 'Gerade eben' +z.string.de.conversation_location_link = 'Zeige Standort' +z.string.de.conversation_member_join = ' hat %@names hinzugefügt' +z.string.de.conversation_member_join_you = ' hast %@names hinzugefügt' +z.string.de.conversation_member_leave_left = ' hat die Unterhaltung verlassen' +z.string.de.conversation_member_leave_left_you = ' hast die Unterhaltung verlassen' +z.string.de.conversation_member_leave_removed = ' hat %@names entfernt' +z.string.de.conversation_member_leave_removed_you = ' hast %@names entfernt' +z.string.de.conversation_rename = ' hat die Unterhaltung umbenannt' +z.string.de.conversation_rename_you = ' hast die Unterhaltung umbenannt' +z.string.de.conversation_resume = 'Beginne eine Unterhaltung mit %@names' +z.string.de.conversation_ping = ' hat gepingt' +z.string.de.conversation_ping_you = ' hast gepingt' +z.string.de.conversation_today = 'Heute' +z.string.de.conversation_verified = 'Überprüft' +z.string.de.conversation_voice_channel_deactivate = ' hat versucht anzurufen' +z.string.de.conversation_voice_channel_deactivate_you = ' hast versucht anzurufen' +z.string.de.conversation_yesterday = 'Gestern' +z.string.de.conversation_unable_to_decrypt_1 = 'eine Nachricht von %@name wurde nicht empfangen.' +z.string.de.conversation_unable_to_decrypt_2 = '%@names Geräte-Identität hat sich geändert. Nachricht kann nicht entschlüsselt werden.' +z.string.de.conversation_unable_to_decrypt_link = 'Warum?' +z.string.de.conversation_unable_to_decrypt_error_message = 'Fehler' +z.string.de.conversation_unable_to_decrypt_reset_session = 'Session zurücksetzen' +z.string.de.conversation_asset_uploading = 'Hochladen…' +z.string.de.conversation_asset_downloading = 'Herunterladen…' +z.string.de.conversation_asset_upload_failed = 'Hochladen fehlgeschlagen' +z.string.de.conversation_asset_upload_too_large = 'Du kannst Dateien bis zu %no senden.' +z.string.de.conversation_playback_error = 'Konnte nicht abgespielt werden' + +# Conversation list +z.string.de.conversation_list_archive = 'Archiv' +z.string.de.conversation_list_empty_conversation = 'Leere Unterhaltung' +z.string.de.conversation_list_many_connection_request = '%no Kontaktanfragen' +z.string.de.conversation_list_one_connection_request = 'Eine Kontaktanfrage' +z.string.de.conversation_list_search_header_placeholder = 'Unterhaltung beginnen' +z.string.de.conversation_list_popover_archive = 'Archivieren' +z.string.de.conversation_list_popover_block = 'Blockieren' +z.string.de.conversation_list_popover_cancel = 'Anfrage abbrechen' +z.string.de.conversation_list_popover_clear = 'Löschen' +z.string.de.conversation_list_popover_leave = 'Verlassen' +z.string.de.conversation_list_popover_notify = 'Benachrichtigen' +z.string.de.conversation_list_popover_silence = 'Stummschalten' +z.string.de.conversation_list_popover_unarchive = 'Dearchivieren' + +# Invites +z.string.de.invite_meta_key_mac = 'Cmd' +z.string.de.invite_meta_key_pc = 'Strg' +z.string.de.invite_hint_selected = 'Zum Kopieren %meta_key + C drücken' +z.string.de.invite_hint_unselected = 'Markieren und %meta_key + C drücken' +z.string.de.invite_detail = 'Der Link gilt für zwei Wochen.' +z.string.de.invite_headline = 'Lade Freunde zu Wire ein' +z.string.de.invite_message = 'Füge mich auf Wire als Kontakt hinzu.\nModerne, private Kommunikation. %url' + +# Extensions +z.string.de.extensions_bubble_button_gif = 'Gif' + +# Extensions Giphy +z.string.de.extensions_giphy_button_ok = 'Senden' +z.string.de.extensions_giphy_button_more = 'Neues Gif' +z.string.de.extensions_giphy_message = '%tag • über giphy.com' +z.string.de.extensions_giphy_no_gifs = 'Ups, kein GIF' +z.string.de.extensions_giphy_random = 'Zufällig' + +# People View +z.string.de.search_open = 'Öffnen' +z.string.de.search_open_group = 'Unterhaltung erstellen' +z.string.de.people_confirm_label = 'Zur Unterhaltung hinzufügen' +z.string.de.people_common_contacts = 'Ihr beide kennt' +z.string.de.people_people = '%no Personen' +z.string.de.people_search_placeholder = 'Nach Namen suchen' +z.string.de.people_everyone_participates = 'Alle deine Kontakte\nsind bereits in\ndieser Unterhaltung.' +z.string.de.people_no_matches = 'Kein passendes Ergebnis.\nSuche nach einen\nanderen Namen.' +z.string.de.people_invite = 'Freunde einladen' +z.string.de.people_share = 'Teile deine Kontakte' +z.string.de.people_bring_your_friends = 'Hole deine Freunde zu Wire' +z.string.de.people_invite_detail = 'Wir verwenden deine Kontaktdaten, um Kontakte für dich zu erstellen. Wir anonymisieren alle Informationen und geben sie nicht weiter.' +z.string.de.people_invite_button_contacts = 'Aus Kontakte' +z.string.de.people_invite_button_gmail = 'Aus Gmail' +z.string.de.people_invite_headline = 'Hole deine Freunde' +z.string.de.people_tabs_details = 'Details' +z.string.de.people_tabs_devices = 'Geräte' +z.string.de.people_tabs_devices_headline = 'Wire gibt jedem Gerät einen einzigartigen Fingerabdruck. Vergleiche diese mit %@.name und überprüfe deine Unterhaltung.' +z.string.de.people_tabs_devices_why_verify = 'Warum sollte ich meine Unterhaltungen verifizieren?' +z.string.de.people_tabs_no_devices_headline = '%@.name benutzt eine ältere Version von Wire. Es werden keine Geräte angezeigt.' +z.string.de.people_tabs_device_detail_all_my_devices = 'Alle meine Geräte anzeigen' +z.string.de.people_tabs_device_detail_device_fingerprint = 'Fingerabdruck des Geräts' +z.string.de.people_tabs_device_detail_headline = 'Überprüfe, ob dieser Fingerabdruck mit dem auf %bold%@.names Gerät%end übereinstimmt.' +z.string.de.people_tabs_device_detail_how_to = 'Wie mache ich das?' +z.string.de.people_tabs_device_detail_reset_session = 'Session zurücksetzen' +z.string.de.people_tabs_device_detail_show_my_device = 'Zeige meinen Fingerabdruck' +z.string.de.people_tabs_device_detail_verified = 'Ok' + +# Add people to conversation share history dialogue +z.string.de.people_add_to_headline = 'Kontakte hinzufügen und Verlauf teilen?' +z.string.de.people_add_to_message = 'Neu hinzugefügte Kontakte können den bisherigen Verlauf sehen. Erstelle eine neue Unterhaltung, um dies zu vermeiden.' + +# Block user +z.string.de.people_block_headline = 'Blockieren?' +z.string.de.people_block_message = '%@.first_name wird dich auf Wire nicht finden können.' + +# Accept a pending connection dialogue +z.string.de.people_connect_headline = 'Annehmen?' +z.string.de.people_connect_message = '%@.first_name wird zu deinen Kontakten hinzugefügt und die Unterhaltung mit ihm geöffnet.' + +# Cancel a pending request +z.string.de.people_cancel_request_headline = 'Kontaktanfrage abbrechen?' +z.string.de.people_cancel_request_message = 'Ziehe die Kontaktanfrage an %@.first_name zurück.' + +# Leave the conversation dialogue +z.string.de.people_leave_headline = 'Unterhaltung verlassen?' +z.string.de.people_leave_message = 'Du wirst keine Nachrichten in dieser Unterhaltung senden oder empfangen können.' + +# Remove from conversation dialogue +z.string.de.people_remove_headline = 'Entfernen?' +z.string.de.people_remove_message = '%@.first_name wird in dieser Unterhaltung keine Nachrichten schicken oder empfangen können.' + +# Unblock user +z.string.de.people_unblock_headline = 'Freigeben?' +z.string.de.people_unblock_message = '%@.first_name wird dich über Wire wieder finden und kontaktieren können. Du empfängst alle Nachrichten die verschickt wurden, während der Kontakt blockiert war.' + +# Button labels for the actions +z.string.de.people_button_add = 'Kontakte hinzufügen' +z.string.de.people_button_block = 'Blockieren' +z.string.de.people_button_cancel = 'Abbrechen' +z.string.de.people_button_connect = 'Hinzufügen' +z.string.de.people_button_continue = 'Weiter' +z.string.de.people_button_create = 'Unterhaltung erstellen' +z.string.de.people_button_ignore = 'Ignorieren' +z.string.de.people_button_leave = 'Verlassen' +z.string.de.people_button_open = 'Unterhaltung öffnen' +z.string.de.people_button_pending = 'Ausstehend' +z.string.de.people_button_profile = 'Profil' +z.string.de.people_button_remove = 'Entfernen' +z.string.de.people_button_unblock = 'Freigeben' +z.string.de.people_button_no = 'Nein' +z.string.de.people_button_yes = 'Ja' + +# Preferences +z.string.de.preferences_headline = 'Einstellungen' +z.string.de.preferences_account = 'Benutzerkonto' +z.string.de.preferences_password = 'Passwort' +z.string.de.preferences_password_reset = 'Passwort zurücksetzen' +z.string.de.preferences_delete = 'Benutzerkonto löschen' +z.string.de.preferences_email_sent = 'E-Mail gesendet' +z.string.de.preferences_delete_info = 'Wir sende eine E-Mail an %email. Klicke den Link, um dein Benutzerkonto dauerhaft zu löschen.' + +z.string.de.preferences_contacts = 'Kontakte' +z.string.de.preferences_share_contacts = 'Aus Gmail importieren' +z.string.de.preferences_share_osx_contacts = 'Aus Kontakte importieren' +z.string.de.preferences_contacts_detail = 'Wir verwenden deine Daten, um Kontakte für dich zu erstellen. Wir anonymisieren alle Informationen und teilen sie mit niemand anderem.' +z.string.de.preferences_data = 'Daten senden' +z.string.de.preferences_data_checkbox = 'Senden von anonymen Nutzungsdaten' +z.string.de.preferences_data_detail = 'Sende uns Fehler- und Nutzungsberichte. Hilf uns Wire durch das Senden anonymisierter Informationen zu verbessern.' +z.string.de.preferences_sound = 'Benachrichtigungen' +z.string.de.preferences_sound_all = 'Alle' +z.string.de.preferences_sound_all_hint = 'Alle Sounds' +z.string.de.preferences_sound_none = 'Keine' +z.string.de.preferences_sound_none_hint = 'Pssst!' +z.string.de.preferences_sound_some = 'Einige' +z.string.de.preferences_sound_some_hint = 'Pings und Anrufe' +z.string.de.preferences_devices = 'Geräte' +z.string.de.preferences_device_remove = 'Gerät entfernen' +z.string.de.preferences_device_remove_detail = 'Entferne dieses Gerät, wenn du es nicht mehr benutzt.' +z.string.de.preferences_device_button_remove = 'Entfernen' +z.string.de.preferences_device_button_cancel = 'Abbrechen' +z.string.de.preferences_device_activated = 'Aktiviert in ' +z.string.de.preferences_device_id = 'ID: ' +z.string.de.preferences_device_reset_session_description = 'Falls die Fingerabdrücke nicht übereinstimmen, setze die Session zurück um sicherzustellen, dass die richtigen Fingerabdrücke auf allen Geräten angezeigt werden.' +z.string.de.preferences_device_reset_session_button = 'Session zurücksetzen' +z.string.de.preferences_device_fingerprint_label = 'Schlüssel-Fingerabdruck' +z.string.de.preferences_device_fingerprint_message = 'Wire gibt jedem Gerät einen einzigartigen Fingerabdruck. Vergleiche die Fingerabdrücke zur Verifizierung deiner Geräte und Unterhaltungen.' + +# Profile +z.string.de.profile_about = 'Über' +z.string.de.profile_preferences = 'Einstellungen' +z.string.de.profile_sign_out = 'Abmelden' +z.string.de.profile_support = 'Hilfe' +z.string.de.profile_username_placeholder = 'Dein vollständiger Name' + +# Search +z.string.de.search_group_hint = 'Tippe weiter oder wähle weitere Kontakte aus, um eine Gruppe zu erstellen' +z.string.de.search_connect = 'Vorschläge' +z.string.de.search_connections = 'Kontakte' +z.string.de.search_groups = 'Gruppen' +z.string.de.search_placeholder = 'Namen oder E-Mail-Adresse suchen' +z.string.de.search_top_people = 'Top Kontakte' +z.string.de.search_try_search = 'Finde Kontakte nach Namen oder\nihrer vollständigen E-Mail-Adresse' +z.string.de.search_no_contacts_on_wire = 'Du hast keine Kontakte auf Wire.\nSuch nach Namen oder vollständigen\nE-Mail-Adressen' +z.string.de.search_others = 'Suchergebnisse' +z.string.de.search_suggestion_one = 'Kennt %@.first_name' +z.string.de.search_suggestion_two = 'Kennt %@.first_name und %@.other_name' +z.string.de.search_suggestion_many = 'Kennt %@.first_name und %no weitere' + +# Picture upload +z.string.de.upload_welcome = 'Wire ist so viel schöner mit einem Bild.' +z.string.de.upload_welcome_keep = 'Behalte dieses' +z.string.de.upload_welcome_choose = 'Wähle dein eigenes' + +# Google contacts upload +z.string.de.upload_google_headline = 'Finde Kontakte \nauf Wire.' +z.string.de.upload_google_message = 'Wir verwenden deine Daten, um Kontakte für dich zu erstellen. Wir anonymisieren alle Informationen und teilen sie mit niemand anderem.' +z.string.de.upload_google_headline_error = 'Ein Fehler ist aufgetreten.' +z.string.de.upload_google_message_error = 'Wir haben die Informationen nicht erhalten. Bitte importiere deine Kontakte erneut.' +z.string.de.upload_google_button_again = 'Erneut versuchen' + +# URLs +z.string.de.url_password_reset = 'https://wire.com/forgot/?hl=de' +z.string.de.url_legal = 'https://wire.com/legal/?hl=de' +z.string.de.url_privacy = 'https://wire.com/privacy/?hl=de' +z.string.de.url_privacy_why = 'https://wire.com/privacy/why/' +z.string.de.url_support = 'https://support.wire.com/hc/de' +z.string.de.url_terms_of_use = 'https://wire.com/legal/terms/' +z.string.de.url_wire = 'https://wire.com/?hl=de' +z.string.de.url_wire_for_web = 'https://app.wire.com' +z.string.de.url_support_calling = 'https://support.wire.com/hc/de/articles/202969412' +z.string.de.url_support_camera_access_denied = 'https://support.wire.com/hc/de/articles/202935412' +z.string.de.url_support_contact_bug = 'https://support.wire.com/hc/de/requests/new?ticket_form_id=101615' +z.string.de.url_support_history = 'https://support.wire.com/hc/de/articles/207834645' +z.string.de.url_support_mic_access_denied = 'https://support.wire.com/hc/de/articles/202590081' +z.string.de.url_support_mic_not_found = 'https://support.wire.com/hc/de/articles/202970662' +z.string.de.url_support_screen_access_denied = 'https://support.wire.com/hc/de/articles/202935412' +z.string.de.url_support_screen_whitelist = 'https://support.wire.com/hc/de/articles/202935412' +z.string.de.url_support_session = 'https://support.wire.com/hc/de' +z.string.de.url_downloads = 'https://wire.com/download/?hl=de' +z.string.de.url_app_store = 'https://wire.com/download/osx/' +z.string.de.url_decrypt_error_1 = 'https://wire.com/privacy/error-1/?hl=de' +z.string.de.url_decrypt_error_2 = 'https://wire.com/privacy/error-2/?hl=de' + +# Warnings: Permission requests & permission callbacks +z.string.de.warning_call_detail = 'Dein Browser benötigt für Anrufe Zugriff auf das Mikrofon.' +z.string.de.warning_call_headline = 'Anrufe sind ohne Mikrofon nicht möglich' +z.string.de.warning_call_unsupported_incoming = '%s.first_name ruft an. Dein Browser unterstützt keine Anrufe.' +z.string.de.warning_call_unsupported_outgoing = 'Du kannst nicht anrufen, da dein Browser keine Anfrufe unterstützt.' +z.string.de.warning_call_issues = 'Diese Version von Wire kann nicht an Anrufen teilnehmen. Nutze' +z.string.de.warning_call_upgrade_browser = 'Für Anrufe aktualisiere Google Chrome.' +z.string.de.warning_learn_more = 'Erfahre mehr' +z.string.de.warning_not_found_camera = 'Du kannst nicht anrufen, da dein Computer keine Kamera hat.' +z.string.de.warning_not_found_microphone = 'Du kannst nicht anrufen, da dein Computer kein Mikrofon hat.' +z.string.de.warning_permission_denied_camera = 'Du kannst nicht anrufen, da dein Browser keinen Zugriff auf die Kamera hat.' +z.string.de.warning_permission_denied_microphone = 'Du kannst nicht anrufen, da dein Browser keinen Zugriff auf das Mikrofon hat.' +z.string.de.warning_permission_denied_screen = 'Du kannst deinen Bildschirm nicht teilen, da dein Browser keinen Zugriff auf den Bildschirm hat.' +z.string.de.warning_permission_request_camera = '%icon Zugriff auf Kamera gewähren' +z.string.de.warning_permission_request_microphone = '%icon Zugriff auf Mikrofon gewähren' +z.string.de.warning_permission_request_notification = '%icon Benachrichtigungen zulassen' +z.string.de.warning_permission_request_screen = '%icon Zugriff auf Bildschirm gewähren' +z.string.de.warning_tell_me_how = 'Zeig mir wie' + +# Warnings: Connectivity +z.string.de.warning_connectivity_connection_lost = 'Verbindung wird wiederhergestellt. Wire kann Nachrichten möglicherweise nicht empfangen.' +z.string.de.warning_connectivity_no_internet = 'Keine Internetverbindung. Du kannst keine Nachrichten senden und empfangen.' + +# Warnings: App banner +z.string.de.warning_app_banner_win = 'Wire ist für Windows verfügbar.' +z.string.de.warning_app_banner_osx = 'Wire ist für OS X verfügbar.' +z.string.de.warning_app_banner_link = 'App herunterladen' + +# Browser notifications +z.string.de.system_notification_asset_add = 'Hat ein Bild geteilt' +z.string.de.system_notification_connection_accepted = 'Hat deine Kontaktanfrage akzeptiert' +z.string.de.system_notification_connection_request = 'Möchte dich als Kontakt hinzufügen' +z.string.de.system_notification_conversation_create = '%s.first_name hat eine Unterhaltung begonnen' +z.string.de.system_notification_conversation_rename = '%s.first_name hat die Unterhaltung in %name umbenannt' +z.string.de.system_notification_member_join_many = '%s.first_name hat %no Kontakte zur Unterhaltung hinzugefügt' +z.string.de.system_notification_member_join_one = '%s.first_name hat %@.first_name zur Unterhaltung hinzugefügt' +z.string.de.system_notification_member_leave_left = '%s.first_name hat die Unterhaltung verlassen' +z.string.de.system_notification_member_leave_removed_many = '%s.first_name hat %no Kontakte aus der Unterhaltung entfernt' +z.string.de.system_notification_member_leave_removed_one = '%s.first_name hat %@.first_name aus der Unterhaltung entfernt' +z.string.de.system_notification_ping = 'Hat gepingt' +z.string.de.system_notification_voice_channel_activate = 'Ruft an' +z.string.de.system_notification_voice_channel_deactivate = 'Hat versucht anzurufen' +z.string.de.system_notification_shared_audio = 'Hat eine Audio-Nachricht geteilt' +z.string.de.system_notification_shared_video = 'Hat ein Video geteilt' +z.string.de.system_notification_shared_file = 'Hat eine Datei geteilt' + +# Tooltips +z.string.de.tooltip_call_banner_accept = 'Anruf annehmen' +z.string.de.tooltip_call_banner_ignore = 'Ignorieren (%shortcut)' +z.string.de.tooltip_call_banner_mute = 'Mikrofon stummschalten (%shortcut)' +z.string.de.tooltip_call_banner_unmute = 'Mikrofon anschalten (%shortcut)' + +z.string.de.tooltip_conversation_call = 'Anruf' +z.string.de.tooltip_conversation_video_call = 'Videoanruf' +z.string.de.tooltip_conversation_file = 'Datei senden' +z.string.de.tooltip_conversation_people = 'Unterhaltungsübersicht (%shortcut)' +z.string.de.tooltip_conversation_picture = 'Bild senden' +z.string.de.tooltip_conversation_ping = 'Ping (%shortcut)' +z.string.de.tooltip_conversation_all_verified = 'Alle Fingerabdrücke sind überprüft' +z.string.de.tooltip_conversation_input_placeholder = 'Schreibe eine Nachricht' + +z.string.de.tooltip_conversation_list_archive = 'Archivieren (%shortcut)' +z.string.de.tooltip_conversation_list_archived = 'Archiv anzeigen (%no)' +z.string.de.tooltip_conversation_list_more = 'Mehr' +z.string.de.tooltip_conversation_list_notify = 'Benachrichtigen (%shortcut)' +z.string.de.tooltip_conversation_list_silence = 'Stummschalten (%shortcut)' +z.string.de.tooltip_conversation_list_tooltip_start = 'Unterhaltung beginnen (%shortcut)' + +z.string.de.tooltip_people_add = 'Kontakte zur Unterhaltung hinzufügen (%shortcut)' +z.string.de.tooltip_people_back = 'Zurück' +z.string.de.tooltip_people_block = 'Blockieren' +z.string.de.tooltip_people_connect = 'Als Kontakt hinzufügen' +z.string.de.tooltip_people_leave = 'Unterhaltung verlassen' +z.string.de.tooltip_people_open = 'Unterhaltung öffnen' +z.string.de.tooltip_people_profile = 'Profil öffnen' +z.string.de.tooltip_people_rename = 'Unterhaltung umbenennen' +z.string.de.tooltip_people_remove = 'Aus Unterhaltung entfernen' +z.string.de.tooltip_people_unblock = 'Freigeben' + +z.string.de.tooltip_preferences_contacts = 'Logge dich in dein Gmail-Konto ein, um deine Kontakte zu teilen' +z.string.de.tooltip_preferences_password = 'Öffne eine andere Website, um dein Passwort zurückzusetzen' +z.string.de.tooltip_preferences_sound = 'Wähle, ob du über neue Nachrichten per Sound informiert wirst' + +z.string.de.tooltip_profile_picture = 'Ändere dein Bild…' +z.string.de.tooltip_profile_preferences = 'Einstellungen…' +z.string.de.tooltip_profile_rename = 'Ändere deinen Namen' + +z.string.de.tooltip_search_close = 'Schließen (Esc)' + +# App loading +z.string.de.init_received_access_token = 'Einloggen' +z.string.de.init_received_self_user = 'Hallo, %name.' +z.string.de.init_sessions_expectation = 'Wir müssen %sessions Sessions initialisieren' +z.string.de.init_sessions_expectation_long = 'Wir müssen %sessions Sessions initialisieren' +z.string.de.init_sessions_progress = 'Initialisiere Sessions - %progress von %total' +z.string.de.init_initialized_storage = 'Sessions geladen' +z.string.de.init_initialized_cryptography = 'Verschlüsselung initialisiert' +z.string.de.init_validated_client = 'Lade deine Kontakte und Unterhaltungen' +z.string.de.init_received_user_data = 'Suche nach neuen Events' +z.string.de.init_events_expectation = 'Du hast %events neue Events' +z.string.de.init_events_expectation_long = 'Lade %events neue Events' +z.string.de.init_events_progress = 'Lade Events - %progress von %total' +z.string.de.init_updated_from_notifications = 'Fast geschafft' +z.string.de.init_app_pre_loaded = 'Viel Spaß mit Wire' diff --git a/app/script/localization/strings-init.coffee b/app/script/localization/strings-init.coffee new file mode 100644 index 00000000000..4dbc41ad383 --- /dev/null +++ b/app/script/localization/strings-init.coffee @@ -0,0 +1,26 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.string ?= {} +z.string.de ?= {} + +z.string.Declension = + ACCUSATIVE: 'accusative' + DATIVE: 'dative' + NOMINATIVE: 'nominative' diff --git a/app/script/localization/strings.coffee b/app/script/localization/strings.coffee new file mode 100644 index 00000000000..c9c04745f48 --- /dev/null +++ b/app/script/localization/strings.coffee @@ -0,0 +1,566 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +#General terms +z.string.wire = 'Wire' +z.string.wire_osx = 'Wire for OS X' +z.string.wire_windows = 'Wire for Windows' +z.string.truncation = '…' +z.string.nonexistent_user = 'Deleted User' +z.string.and = 'and' + +# About screen +z.string.about_copyright = '© Wire Swiss GmbH • Version' +z.string.about_legal = 'Legal' +z.string.about_privacy = 'Privacy' +z.string.about_wire = 'wire.com' + +# Alert view when trying to set a profile image that's too small +z.string.alert_upload_file_format = 'Can’t use this picture.\nPlease choose a PNG or JPEG file.' +z.string.alert_upload_too_small = 'Can’t use this picture.\nPlease choose a picture that’s at least 320 x 320 px.' +z.string.alert_upload_too_large = 'This picture is too large.\nYou can upload files up to %no MB.' +z.string.alert_gif_too_large = 'Animation is too large.\nMaximum size is %no MB.' + +# Auth +# Authentication: ACCOUNT section +z.string.auth_account_country_code = 'Country Code' +z.string.auth_account_create = 'Create' +z.string.auth_account_create_account = 'Create an account' +z.string.auth_account_expiration = 'You were signed out because your session expired. Please log in again.' +z.string.auth_account_get_wire = 'Simple, private & secure messenger for chat, calls, sharing pics, music, videos, GIFs and more.' +z.string.auth_account_password_forgot = 'Forgot password' +z.string.auth_account_remember_me = 'Remember me' +z.string.auth_account_sign_in = 'Log in' +z.string.auth_account_sign_in_email = 'Email' +z.string.auth_account_sign_in_phone = 'Phone' +z.string.auth_account_terms_of_use = 'Terms of Use' +z.string.auth_account_terms_of_use_detail = 'I accept' + +# Authentication: VERIFY section +z.string.auth_verify_code_description = 'Enter the verification code\nwe sent to %@number.' +z.string.auth_verify_code_resend = 'No code showing up?' +z.string.auth_verify_code_resend_detail = 'Resend' +z.string.auth_verify_code_resend_timer = 'You can request a new code %expiration.' +z.string.auth_verify_code_change_phone = 'Change phone number' +z.string.auth_verify_email_button = 'Add' +z.string.auth_verify_email_detail = 'Using Wire in multiple devices requires an email address and password.' +z.string.auth_verify_email_headline = 'Hello, %name.' + +# Authentication: limit section +z.string.auth_limit_devices_headline = 'Devices' +z.string.auth_limit_description = 'Remove one of your other devices to start using Wire on this one.' +z.string.auth_limit_button_manage = 'Manage devices' +z.string.auth_limit_button_sign_out = 'Log out' +z.string.auth_limit_devices_current = '(Current)' + +# Authentication: limit section +z.string.auth_history_headline = 'It’s the first time you’re using Wire on this device.' +z.string.auth_history_description = 'For privacy reasons, your conversation history will not appear here.' +z.string.auth_history_button = 'OK' + +# Authentication: POSTED section +z.string.auth_posted_change_email = 'Change email' +z.string.auth_posted_offline_detail = 'Check your Internet connection and try again.' +z.string.auth_posted_offline_headline = 'Wire is nicer online.' +z.string.auth_posted_pending_detail = 'Check your email inbox or resend activation.' +z.string.auth_posted_pending_headline = 'Account already pending' +z.string.auth_posted_resend = 'Resend to %email' +z.string.auth_posted_resend_action = 'No email showing up?' +z.string.auth_posted_resend_detail = 'Check your email inbox and follow the instructions.' +z.string.auth_posted_resend_headline = 'You’ve got mail.' +z.string.auth_posted_retry = 'Re-try and send mail to %email' +z.string.auth_posted_retry_action = 'Try again?' +z.string.auth_posted_retry_detail = 'Please try again.' +z.string.auth_posted_retry_headline = 'Something went wrong' +z.string.auth_posted_verify_later = 'Verify later' + +#Authentication: Misc +z.string.auth_placeholder_email = 'Email' +z.string.auth_placeholder_name = 'Name' +z.string.auth_placeholder_password_put = 'Password' +z.string.auth_placeholder_password_set = 'Password (at least 8 characters)' +z.string.auth_placeholder_phone = 'Phone Number' +z.string.auth_hello = 'Hello, %name.' + +# Authentication: Validation errors +z.string.auth_error_code = 'Invalid Code' +z.string.auth_error_country_code_invalid = 'Invalid Country Code' +z.string.auth_error_email_exists = 'Email address already taken' +z.string.auth_error_email_forbidden = 'Sorry. This email address is forbidden.' +z.string.auth_error_email_malformed = 'Please enter a valid email address.' +z.string.auth_error_email_missing = 'Please enter an email address.' +z.string.auth_error_misc = 'Problems with the connection. Please try again.' +z.string.auth_error_name_long = 'Please enter a shorter name' +z.string.auth_error_name_short = 'Enter a name with at least 2 characters' +z.string.auth_error_offline = 'No Internet connection' +z.string.auth_error_password_long = 'The password you entered is too long' +z.string.auth_error_password_short = 'Choose a password with at least 8 characters.' +z.string.auth_error_password_wrong = 'Wrong password. Please try again.' +z.string.auth_error_phone_number_invalid = 'Invalid Phone Number' +z.string.auth_error_phone_number_unknown = 'Unknown Phone Number' +z.string.auth_error_sign_in = 'Please verify your details and try again.' +z.string.auth_error_terms_of_use = 'Please accept Wire Terms of Use.' + +# Call stuff +z.string.call_state_outgoing = 'Ringing…' +z.string.call_state_connecting = 'Connecting…' +z.string.call_state_incoming = 'Calling…' +z.string.call_decline = 'Decline' +z.string.call_accept = 'Accept' +z.string.call_join = 'Join' +z.string.call_choose_shared_screen = 'Choose a screen to share' + +# Calling Bar +z.string.call_banner_connecting = 'Connecting…' +z.string.call_banner_in = 'in' +z.string.call_banner_incoming_1to1 = '%s.first_name calling' +z.string.call_banner_incoming_video_1to1 = '%s.first_name video calling' +z.string.call_banner_incoming_group = '%s.first_name calling %@.group' +z.string.call_banner_join = 'Join call' +z.string.call_banner_ongoing = 'Ongoing call' +z.string.call_banner_outgoing = 'Calling %@.first_name' + +# Warnings +z.string.modal_button_cancel = 'Cancel' +z.string.modal_button_ok = 'Ok' + +# Block a user +z.string.modal_block_conversation_headline = 'Block %@.name?' +z.string.modal_block_conversation_message = '%@.name wont be able to contact you or invite you to a group conversation.' +z.string.modal_block_conversation_button = 'Block' +# Cannot create the call because there is nobody to call (conversation_empty) +z.string.modal_call_conversation_empty_headline = 'No one to call' +z.string.modal_call_conversation_empty_message = 'There is no one left here.' +# Cannot create the call because there are too many participants (conversation_full) +z.string.modal_call_conversation_full_headline = 'Too many people to call' +z.string.modal_call_conversation_full_message = 'Calls work in conversations with up to %no people.' +# Cannot video call in group conversations +z.string.modal_call_no_video_in_group_headline = 'No video calls in groups' +z.string.modal_call_no_video_in_group_message = 'Video calls are not available in group conversations.' +# Second incoming call +z.string.modal_call_second_incoming_headline = 'Answer call?' +z.string.modal_call_second_incoming_message = 'Your current call will end.' +z.string.modal_call_second_incoming_action = 'Answer' +# Second outgoing call +z.string.modal_call_second_outgoing_headline = 'Hang up current call?' +z.string.modal_call_second_outgoing_message = 'You can only be in one call at a time.' +z.string.modal_call_second_outgoing_action = 'Hang Up' +# Cannot join the call because there are too many participants (voice_channel_full) +z.string.modal_call_voice_channel_full_headline = 'Full house' +z.string.modal_call_voice_channel_full_message = 'There’s only room for %no people in here.' +# Clear a conversation +z.string.modal_clear_conversation_headline = 'Delete "%@.name" content?' +z.string.modal_clear_conversation_message = 'This will clear the conversation history and remove it from your list.' +z.string.modal_clear_conversation_option = 'Also leave the conversation' +z.string.modal_clear_conversation_button = 'Delete' +# Connected device +z.string.modal_connected_device_headline = 'Your account was used on:' +z.string.modal_connected_device_from = 'From:' +z.string.modal_connected_device_message = 'If you didn’t do this, remove the device and reset your password.' +z.string.modal_connected_device_manage_devices = 'manage devices' +# Delete message +z.string.modal_delete_button = 'Delete' +z.string.modal_delete_headline = 'Delete message' +z.string.modal_delete_message = 'The message will only be removed from your view of the conversation. This cannot be undone.' +# Too long message +z.string.modal_too_long_headline = 'Message too long' +z.string.modal_too_long_message = 'You can send messages up to %no characters long.' +# Leave a conversation +z.string.modal_leave_conversation_headline = 'Leave "%@.name" conversation?' +z.string.modal_leave_conversation_message = 'The participants will be notified and the conversation removed from your list.' +z.string.modal_leave_conversation_button = 'Leave' +# Logout +z.string.modal_logout_headline = 'Clear Data?' +z.string.modal_logout_message = 'Delete all your personal information and conversations on this device.' +z.string.modal_logout_button = 'Log out' +# New device +z.string.modal_new_device_headline = '"%@.name" started using a new device' +z.string.modal_new_device_message = 'Do you still want to send your messages?' +z.string.modal_new_device_show_device = 'show device' +z.string.modal_new_device_send_anyway = 'send anyway' +# Session Reset +z.string.modal_session_reset_headline = 'The session has been reset' +z.string.modal_session_reset_message_1 = 'If the problem is not resolved,' +z.string.modal_session_reset_message_link = 'contact' +z.string.modal_session_reset_message_2 = 'us.' +# Too many members in conversation +z.string.modal_too_many_members_headline = 'Full house' +z.string.modal_too_many_members_message = 'Up to %max people can join a conversation. There is room for %no more people in here.' +# Whitelist screensharing +z.string.modal_whitelist_screensharing_headline = 'Whitelist us for screen sharing' +z.string.modal_whitelist_screensharing_message_1 = 'Until Firefox adds us to the list, manually add the current domain on "about:config" to "media.getusermedia.screensharing.allowed_domains". Follow the' +z.string.modal_whitelist_screensharing_message_link = 'FAQ' +z.string.modal_whitelist_screensharing_message_2 = 'that guides you through the process.' +# Parallel uploads +z.string.modal_uploads_parallel = 'You can send up to %no files at once.' + +# Connection requests +z.string.connection_request_connect = 'Connect' +z.string.connection_request_ignore = 'Ignore' +z.string.connection_request_message = 'Hi %@.first_name,\nLet’s connect on Wire.\n%s.first_name' + +# Conversation +z.string.conversation_you_nominative = 'you' +z.string.conversation_you_dative = 'you' +z.string.conversation_you_accusative = 'you' + +z.string.conversation_connection_accepted = 'Connected' +z.string.conversation_connection_cancel_request = 'Cancel connection request' +z.string.conversation_connection_blocked = 'Blocked' +z.string.conversation_connection_pending = 'Connecting' +z.string.conversation_create = ' started a conversation with %@names' +z.string.conversation_create_you = ' started a conversation with %@names' +z.string.conversation_device_started_using = ' started using' +z.string.conversation_device_started_using_you = ' started using' +z.string.conversation_device_unverified = ' unverified one of' +z.string.conversation_device_your_devices = ' your devices' +z.string.conversation_device_user_devices = ' %@name´s devices' +z.string.conversation_device_a_new_device = ' a new device' +z.string.conversation_device_this_device = ' this device' +z.string.conversation_just_now = 'Just now' +z.string.conversation_location_link = 'Open Map' +z.string.conversation_member_join = ' added %@names' +z.string.conversation_member_join_you = ' added %@names' +z.string.conversation_member_leave_left = ' left' +z.string.conversation_member_leave_left_you = ' left' +z.string.conversation_member_leave_removed = ' removed %@names' +z.string.conversation_member_leave_removed_you = ' removed %@names' +z.string.conversation_rename = ' renamed the conversation' +z.string.conversation_rename_you = ' renamed the conversation' +z.string.conversation_resume = 'Start a conversation with %@names' +z.string.conversation_ping = ' pinged' +z.string.conversation_ping_you = ' pinged' +z.string.conversation_today = 'today' +z.string.conversation_verified = 'Verified' +z.string.conversation_voice_channel_deactivate = ' called' +z.string.conversation_voice_channel_deactivate_you = ' called' +z.string.conversation_yesterday = 'Yesterday' +z.string.conversation_unable_to_decrypt_1 = 'a message from %@name was not received.' +z.string.conversation_unable_to_decrypt_2 = '%@name´s device identity changed. Undelivered message.' +z.string.conversation_unable_to_decrypt_link = 'Why?' +z.string.conversation_unable_to_decrypt_error_message = 'Error' +z.string.conversation_unable_to_decrypt_reset_session = 'Reset session' +z.string.conversation_asset_uploading = 'Uploading…' +z.string.conversation_asset_downloading = 'Downloading…' +z.string.conversation_asset_upload_failed = 'Upload Failed' +z.string.conversation_asset_upload_too_large = 'You can send files up to %no' +z.string.conversation_playback_error = 'Unable to play' + +# Conversation list +z.string.conversation_list_archive = 'ARCHIVE' +z.string.conversation_list_empty_conversation = 'Empty conversation' +z.string.conversation_list_many_connection_request = '%no people waiting' +z.string.conversation_list_one_connection_request = '1 person waiting' +z.string.conversation_list_search_header_placeholder = 'Start a conversation' +z.string.conversation_list_popover_archive = 'Archive' +z.string.conversation_list_popover_block = 'Block' +z.string.conversation_list_popover_cancel = 'Cancel request' +z.string.conversation_list_popover_clear = 'Delete' +z.string.conversation_list_popover_leave = 'Leave' +z.string.conversation_list_popover_notify = 'Unmute' +z.string.conversation_list_popover_silence = 'Mute' +z.string.conversation_list_popover_unarchive = 'Unarchive' + +# Invites +z.string.invite_meta_key_mac = 'Cmd' +z.string.invite_meta_key_pc = 'Ctrl' +z.string.invite_hint_selected = 'Press %meta_key + C to copy' +z.string.invite_hint_unselected = 'Select and Press %meta_key + C' +z.string.invite_detail = 'The link can be used for 2 weeks.' +z.string.invite_headline = 'Invite people to Wire' +z.string.invite_message = 'Let’s connect on Wire.\nModern, private communication. %url' + +# Extensions +z.string.extensions_bubble_button_gif = 'Gif' + +# Extensions Giphy +z.string.extensions_giphy_button_ok = 'Send' +z.string.extensions_giphy_button_more = 'Try Another' +z.string.extensions_giphy_message = '%tag • via giphy.com' +z.string.extensions_giphy_no_gifs = 'Oops, no gifs' +z.string.extensions_giphy_random = 'Random' + +# People View +z.string.search_open = 'Open' +z.string.search_open_group = 'Create Group' +z.string.people_confirm_label = 'Add to conversation' +z.string.people_common_contacts = 'You both know' +z.string.people_people = '%no People' +z.string.people_search_placeholder = 'Search by name' +z.string.people_everyone_participates = 'Everyone you’re\nconnected to is already in\nthis conversation.' +z.string.people_no_matches = 'No matching results.\nTry entering a different name.' +z.string.people_invite = 'Invite people' +z.string.people_share = 'Share Contacts' +z.string.people_bring_your_friends = 'Bring your Friends to Wire' +z.string.people_invite_detail = 'Sharing your contacts helps you connect with others. We anonymize all the information and do not share it with anoyone else.' +z.string.people_invite_button_contacts = 'From Contacts' +z.string.people_invite_button_gmail = 'From Gmail' +z.string.people_invite_headline = 'Bring your friends' +z.string.people_tabs_details = 'Details' +z.string.people_tabs_devices = 'Devices' +z.string.people_tabs_devices_headline = 'Wire gives every device a unique fingerprint. Compare them with %@.name and verify your conversation.' +z.string.people_tabs_devices_why_verify = 'Why verify conversation?' +z.string.people_tabs_no_devices_headline = '%@.name is using an old version of Wire. No devices are shown here.' +z.string.people_tabs_device_detail_all_my_devices = 'Show all my devices' +z.string.people_tabs_device_detail_device_fingerprint = 'Device fingerprint' +z.string.people_tabs_device_detail_headline = 'Verify that this matches the fingerprint shown on %bold%@.name’s device%end.' +z.string.people_tabs_device_detail_how_to = 'How do I do that?' +z.string.people_tabs_device_detail_reset_session = 'Reset session' +z.string.people_tabs_device_detail_show_my_device = 'Show my device fingerprint' +z.string.people_tabs_device_detail_verified = 'Verified' + +# Add people to conversation share history dialogue +z.string.people_add_to_headline = 'Add people and share history?' +z.string.people_add_to_message = 'They will be able to see previous messages. Create a new conversation to avoid that.' + +# Block user +z.string.people_block_headline = 'Block?' +z.string.people_block_message = '%@.first_name won’t be able to find or contact you on Wire.' + +# Accept a pending connection dialogue +z.string.people_connect_headline = 'Accept?' +z.string.people_connect_message = 'This will connect you and open the conversation with %@.first_name.' + +# Cancel a pending request +z.string.people_cancel_request_headline = 'Cancel Request?' +z.string.people_cancel_request_message = 'Remove connection request to %@.first_name.' + +# Leave the conversation dialogue +z.string.people_leave_headline = 'Leave the conversation?' +z.string.people_leave_message = 'You won’t be able to send or receive messages in this conversation.' + +# Remove from conversation dialogue +z.string.people_remove_headline = 'Remove?' +z.string.people_remove_message = '%@.first_name won’t be able to send or receive messages in this conversation.' + +# Unblock user +z.string.people_unblock_headline = 'Unblock?' +z.string.people_unblock_message = '%@.first_name will be able to find and contact you on Wire again. You’ll also receive messages they sent while blocked.' + +# Button labels for the actions +z.string.people_button_add = 'Add people' +z.string.people_button_block = 'Block' +z.string.people_button_cancel = 'Cancel' +z.string.people_button_connect = 'Connect' +z.string.people_button_continue = 'Continue' +z.string.people_button_create = 'Create group' +z.string.people_button_ignore = 'Ignore' +z.string.people_button_leave = 'Leave' +z.string.people_button_open = 'Open Conversation' +z.string.people_button_pending = 'Pending' +z.string.people_button_profile = 'Profile' +z.string.people_button_remove = 'Remove' +z.string.people_button_unblock = 'Unblock' +z.string.people_button_no = 'No' +z.string.people_button_yes = 'Yes' + +# Preferences +z.string.preferences_headline = 'Settings' +z.string.preferences_account = 'Account' +z.string.preferences_password = 'Password' +z.string.preferences_password_reset = 'Reset Password' +z.string.preferences_delete = 'Delete Account' +z.string.preferences_email_sent = 'Email Sent' +z.string.preferences_delete_info = 'We will send an email to: %email follow the link to permanently delete your account.' + +z.string.preferences_contacts = 'Contacts' +z.string.preferences_share_contacts = 'Import from Gmail' +z.string.preferences_share_osx_contacts = 'Import from Contacts' +z.string.preferences_contacts_detail = 'We use your contact data to connect you with others. We anonymize all information and do not share it with anyone else.' +z.string.preferences_data = 'Data' +z.string.preferences_data_checkbox = 'Send anonymous usage data' +z.string.preferences_data_detail = 'Send usage data and crash reports. Make Wire better by sending anonymous information.' +z.string.preferences_sound = 'Sound alerts' +z.string.preferences_sound_all = 'All' +z.string.preferences_sound_all_hint = 'All sounds' +z.string.preferences_sound_none = 'None' +z.string.preferences_sound_none_hint = 'Sshhh!' +z.string.preferences_sound_some = 'Some' +z.string.preferences_sound_some_hint = 'Pings and calls' +z.string.preferences_devices = 'Devices' +z.string.preferences_device_remove = 'Remove Device' +z.string.preferences_device_remove_detail = 'Remove this device if you have stopped using it.' +z.string.preferences_device_button_remove = 'Remove' +z.string.preferences_device_button_cancel = 'Cancel' +z.string.preferences_device_activated = 'Activated in ' +z.string.preferences_device_id = 'ID: ' +z.string.preferences_device_reset_session_description = 'If fingerprints don’t match, reset your secure session to make sure that the correct fingerprints are shown on all devices.' +z.string.preferences_device_reset_session_button = 'Reset session' +z.string.preferences_device_fingerprint_label = 'Key fingerprint' +z.string.preferences_device_fingerprint_message = 'Wire gives every device a unique fingerprint. Compare fingerprints to verify your devices and conversations.' + +# Profile +z.string.profile_about = 'About' +z.string.profile_preferences = 'Settings' +z.string.profile_sign_out = 'Log out' +z.string.profile_support = 'Support' +z.string.profile_username_placeholder = 'Your full name' + +# Search +z.string.search_group_hint = 'Keep typing or pick more people to create a group' +z.string.search_connect = 'Connect' +z.string.search_connections = 'Connections' +z.string.search_groups = 'Groups' +z.string.search_placeholder = 'Search by name or email' +z.string.search_top_people = 'Top people' +z.string.search_try_search = 'Find people by name or\nfull email address' +z.string.search_no_contacts_on_wire = 'You have no contacts on Wire.\nTry finding people by name or\nfull email address' +z.string.search_others = 'Connect' +z.string.search_suggestion_one = 'Knows %@.first_name' +z.string.search_suggestion_two = 'Knows %@.first_name and %@.other_name' +z.string.search_suggestion_many = 'Knows %@.first_name and %no others' + +# Picture upload +z.string.upload_welcome = 'Wire is so much nicer with a picture.' +z.string.upload_welcome_keep = 'keep this one' +z.string.upload_welcome_choose = 'choose your own' + +# Google contacts upload +z.string.upload_google_headline = 'Find people\nto talk to.' +z.string.upload_google_message = 'We use your contact data to connect you with others. We anonymize all information and do not share it with anyone else.' +z.string.upload_google_headline_error = 'Something\nwent wrong.' +z.string.upload_google_message_error = 'We did not receive your information. Please try importing your contacts again.' +z.string.upload_google_button_again = 'Try again' + +# URLs +z.string.url_password_reset = 'https://wire.com/forgot/' +z.string.url_legal = 'https://wire.com/legal/' +z.string.url_privacy = 'https://wire.com/privacy/' +z.string.url_privacy_why = 'https://wire.com/privacy/why/' +z.string.url_support = 'https://support.wire.com' +z.string.url_terms_of_use = 'https://wire.com/legal/terms/' +z.string.url_wire = 'https://wire.com' +z.string.url_wire_for_web = 'https://app.wire.com' +z.string.url_support_calling = 'https://support.wire.com/hc/en-us/articles/202969412' +z.string.url_support_camera_access_denied = 'https://support.wire.com/hc/en-us/articles/202935412' +z.string.url_support_contact_bug = 'https://support.wire.com/hc/en-us/requests/new?ticket_form_id=101615' +z.string.url_support_history = 'https://support.wire.com/hc/en-us/articles/207834645' +z.string.url_support_mic_access_denied = 'https://support.wire.com/hc/en-us/articles/202590081' +z.string.url_support_mic_not_found = 'https://support.wire.com/hc/en-us/articles/202970662' +z.string.url_support_screen_access_denied = 'https://support.wire.com/hc/en-us/articles/202935412' +z.string.url_support_screen_whitelist = 'https://support.wire.com/hc/en-us/articles/202935412' +z.string.url_support_session = 'https://support.wire.com/hc/en-us/' +z.string.url_downloads = 'https://wire.com/download/' +z.string.url_app_store = 'https://wire.com/download/osx/' +z.string.url_decrypt_error_1 = 'https://wire.com/privacy/error-1' +z.string.url_decrypt_error_2 = 'https://wire.com/privacy/error-2' + +# Warnings: Permission requests & permission callbacks +z.string.warning_call_detail = 'Your browser needs access to the microphone to make calls.' +z.string.warning_call_headline = 'Can’t call without microphone' +z.string.warning_call_unsupported_incoming = '%s.first_name is calling. Your browser doesn’t support calls.' +z.string.warning_call_unsupported_outgoing = 'You cannot call because your browser doesn’t support calls.' +z.string.warning_call_issues = 'This version of Wire can not participate in the call. Please use' +z.string.warning_call_upgrade_browser = 'To call, please update Google Chrome.' +z.string.warning_learn_more = 'Learn more' +z.string.warning_not_found_camera = 'You cannot call because your computer does not have a camera.' +z.string.warning_not_found_microphone = 'You cannot call because your computer does not have a microphone.' +z.string.warning_permission_denied_camera = 'You cannot call because your browser does not have access to the camera.' +z.string.warning_permission_denied_microphone = 'You cannot call because your browser does not have access to the microphone.' +z.string.warning_permission_denied_screen = 'You cannot share your screen because the browser does not have access it.' +z.string.warning_permission_request_camera = '%icon Allow access to camera' +z.string.warning_permission_request_microphone = '%icon Allow access to microphone' +z.string.warning_permission_request_notification = '%icon Allow notifications' +z.string.warning_permission_request_screen = '%icon Allow access to screen' +z.string.warning_tell_me_how = 'Tell me how' + +# Warnings: Connectivity +z.string.warning_connectivity_connection_lost = 'Trying to connect. Wire may not be able to deliver messages.' +z.string.warning_connectivity_no_internet = 'No Internet. You won’t be able to send or receive messages.' + +# Warnings: App banner +z.string.warning_app_banner_win = 'Wire is available for Windows.' +z.string.warning_app_banner_osx = 'Wire is available for OS X.' +z.string.warning_app_banner_link = 'Get the app' + +# Browser notifications +z.string.system_notification_asset_add = 'Shared a picture' +z.string.system_notification_connection_accepted = 'Accepted your connection request' +z.string.system_notification_connection_request = 'Wants to connect' +z.string.system_notification_conversation_create = '%s.first_name started a conversation' +z.string.system_notification_conversation_rename = '%s.first_name renamed the conversation to %name' +z.string.system_notification_member_join_many = '%s.first_name added %no people to the conversation' +z.string.system_notification_member_join_one = '%s.first_name added %@.first_name to the conversation' +z.string.system_notification_member_leave_left = '%s.first_name left the conversation' +z.string.system_notification_member_leave_removed_many = '%s.first_name removed %no people from the conversation' +z.string.system_notification_member_leave_removed_one = '%s.first_name removed %@.first_name from the conversation' +z.string.system_notification_ping = 'Pinged' +z.string.system_notification_voice_channel_activate = 'Calling' +z.string.system_notification_voice_channel_deactivate = 'Called' +z.string.system_notification_shared_audio = 'Shared an audio message' +z.string.system_notification_shared_video = 'Shared a video' +z.string.system_notification_shared_file = 'Shared a file' + +# Tooltips +z.string.tooltip_call_banner_accept = 'Accept call' +z.string.tooltip_call_banner_ignore = 'Ignore call (%shortcut)' +z.string.tooltip_call_banner_mute = 'Mute microphone (%shortcut)' +z.string.tooltip_call_banner_unmute = 'Unmute microphone (%shortcut)' + +z.string.tooltip_conversation_call = 'Call' +z.string.tooltip_conversation_video_call = 'Video Call' +z.string.tooltip_conversation_file = 'Add file' +z.string.tooltip_conversation_people = 'People (%shortcut)' +z.string.tooltip_conversation_picture = 'Add picture' +z.string.tooltip_conversation_ping = 'Ping (%shortcut)' +z.string.tooltip_conversation_all_verified = 'All fingerprints are verified' +z.string.tooltip_conversation_input_placeholder = 'Type a message' + +z.string.tooltip_conversation_list_archive = 'Archive (%shortcut)' +z.string.tooltip_conversation_list_archived = 'Show archive (%no)' +z.string.tooltip_conversation_list_more = 'More' +z.string.tooltip_conversation_list_notify = 'Unmute (%shortcut)' +z.string.tooltip_conversation_list_silence = 'Mute (%shortcut)' +z.string.tooltip_conversation_list_tooltip_start = 'Start conversation (%shortcut)' + +z.string.tooltip_people_add = 'Add people to conversation (%shortcut)' +z.string.tooltip_people_back = 'Back' +z.string.tooltip_people_block = 'Block' +z.string.tooltip_people_connect = 'Connect' +z.string.tooltip_people_leave = 'Leave conversation' +z.string.tooltip_people_open = 'Open conversation' +z.string.tooltip_people_profile = 'Open your profile' +z.string.tooltip_people_rename = 'Change conversation name' +z.string.tooltip_people_remove = 'Remove from conversation' +z.string.tooltip_people_unblock = 'Unblock' + +z.string.tooltip_preferences_contacts = 'Sign in to your Gmail account to share contacts' +z.string.tooltip_preferences_password = 'Open another website to reset your password' +z.string.tooltip_preferences_sound = 'Select if events play sounds' + +z.string.tooltip_profile_picture = 'Change your picture…' +z.string.tooltip_profile_preferences = 'Preferences…' +z.string.tooltip_profile_rename = 'Change your name' + +z.string.tooltip_search_close = 'Close (Esc)' + +# App loading +z.string.init_received_access_token = 'Access granted' +z.string.init_received_self_user = 'Hello, %name.' +z.string.init_sessions_expectation = 'We need to initialize %sessions sessions' +z.string.init_sessions_expectation_long = 'We need to initialize %sessions sessions' +z.string.init_sessions_progress = 'Initializing sessions - %progress of %total' +z.string.init_initialized_storage = 'Sessions loaded' +z.string.init_initialized_cryptography = 'Cryptography has been fully set up' +z.string.init_validated_client = 'Loading your connections and your conversations' +z.string.init_received_user_data = 'Checking for new messages' +z.string.init_events_expectation = 'You have %events new messages' +z.string.init_events_expectation_long = 'Loading %events new messages' +z.string.init_events_progress = 'Loading messages - %progress of %total' +z.string.init_updated_from_notifications = 'We are almost there' +z.string.init_app_pre_loaded = 'Enjoy Wire' diff --git a/app/script/location/GeoLocation.coffee b/app/script/location/GeoLocation.coffee new file mode 100644 index 00000000000..31acee0ed57 --- /dev/null +++ b/app/script/location/GeoLocation.coffee @@ -0,0 +1,83 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} + +z.location = do -> + GOOGLE_GEOCODING_BASE_URL = 'https://maps.googleapis.com/maps/api/geocode/json' + API_KEY = 'AIzaSyCKxxKw5JBZ5zEFtoirtgnw8omvH7gWzfo' + + _parse_results = (results) -> + res = {} + result = results[0] + + res['address'] = result.formatted_address + res['lat'] = result.geometry.location.lat + res['lng'] = result.geometry.location.lng + + for component in result.address_components + name = component.long_name or component.short_name + for type in component.types + res[type] = name + if type is 'country' + res['country_code'] = component.short_name or '' + + res['place'] = res.locality or res.natural_feature or res.administrative_area_level_3 or res.administrative_area_level_2 or res.administrative_area_level_1 + delete res.political? + return res + + ### + Reverse loop up for geo location + + @param lat [Number] latitude + @param lng [Number] longitude + @param callback [Function] Function to be called on return + ### + get_location = (lat, lng, callback) -> + if not lat? or not lng? + callback? 'You need to specify latitude and longitude in order to get the location' + + $.ajax + url: "#{GOOGLE_GEOCODING_BASE_URL}?latlng=#{lat},#{lng}&key=#{API_KEY}" + .done (response) -> + if response.status is 'OK' + callback? null, _parse_results response.results + else + callback? response.status + .fail (jqXHR, textStatus, errorThrown) -> + callback? errorThrown + + ### + Return link to google maps + + @param lat [Number] latitude + @param lng [Number] longitude + @param name [String] location name + @param zoom [String] map zoom level + ### + get_maps_url = (lat, lng, name, zoom) -> + base_url = 'https://google.com/maps/' + base_url += "place/#{name}/" if name? + base_url += "@#{lat},#{lng}" + base_url += ",#{zoom}z" if zoom? + return base_url + + return { + get_location: get_location + get_maps_url: get_maps_url + } diff --git a/app/script/main/app.coffee b/app/script/main/app.coffee new file mode 100644 index 00000000000..f356d4bb22f --- /dev/null +++ b/app/script/main/app.coffee @@ -0,0 +1,483 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.main ?= {} + +# @formatter:off +class z.main.App + ### + Construct a new app. + @param auth [z.main.Auth] Authentication settings + ### + constructor: (@auth) -> + @logger = new z.util.Logger 'z.main.App', z.config.LOGGER.OPTIONS + + @telemetry = new z.telemetry.app_init.AppInitTelemetry() + @window_handler = new z.ui.WindowHandler().init() + + @service = @_setup_services() + @repository = @_setup_repositories() + @view = @_setup_view_models() + @util = @_setup_utils() + + @_subscribe_to_events() + + @init_debugging() + @init_app() + + + ############################################################################### + # Instantiation + ############################################################################### + + # Create all app services. + _setup_services: -> + service = {} + + service.asset = new z.assets.AssetService @auth.client + service.call = new z.calling.CallService @auth.client + service.connect = new z.connect.ConnectService @auth.client + service.connect_google = new z.connect.ConnectGoogleService @auth.client + service.cryptography = new z.cryptography.CryptographyService @auth.client + service.giphy = new z.extension.GiphyService @auth.client + service.search = new z.search.SearchService @auth.client + service.storage = new z.storage.StorageService() + service.user = new z.user.UserService @auth.client + service.web_socket = new z.event.WebSocketService @auth.client + + service.client = new z.client.ClientService @auth.client, service.storage + service.conversation = new z.conversation.ConversationService @auth.client, service.storage + service.notification = new z.event.NotificationService @auth.client, service.storage + service.announce = new z.announce.AnnounceService() + + return service + + # Create all app repositories. + _setup_repositories: -> + repository = {} + + repository.audio = @auth.audio + repository.storage = new z.storage.StorageRepository @service.storage + repository.cache = new z.cache.CacheRepository() + repository.cryptography = new z.cryptography.CryptographyRepository @service.cryptography, repository.storage + repository.giphy = new z.extension.GiphyRepository @service.giphy + + repository.client = new z.client.ClientRepository @service.client, repository.cryptography + repository.user = new z.user.UserRepository @service.user, @service.asset, @service.search, repository.client, repository.cryptography + repository.connect = new z.connect.ConnectRepository @service.connect, @service.connect_google, repository.user + repository.event = new z.event.EventRepository @service.web_socket, @service.notification, repository.cryptography, repository.user + repository.search = new z.search.SearchRepository @service.search, repository.user + repository.announce = new z.announce.AnnounceRepository @service.announce + repository.links = new z.links.LinkPreviewRepository @service.asset + + repository.conversation = new z.conversation.ConversationRepository( + @service.conversation, + @service.asset, + repository.user, + repository.giphy, + repository.cryptography, + repository.links + ) + + repository.call_center = new z.calling.CallCenter @service.call, repository.conversation, repository.user, repository.audio + repository.event_tracker = new z.tracking.EventTrackingRepository repository.user, repository.conversation + repository.system_notification = new z.SystemNotification.SystemNotificationRepository repository.conversation + + return repository + + # Create all app view models. + _setup_view_models: -> + view = {} + + view.main = new z.ViewModel.MainViewModel 'wire-main', @repository.user + view.content = new z.ViewModel.RightViewModel 'right', @repository.user, @repository.conversation, @repository.call_center, @repository.search, @repository.giphy, @repository.client + view.background = new z.ViewModel.BackgroundViewModel 'background', view.content, @repository.conversation, @repository.user + view.conversation_list = new z.ViewModel.ConversationListViewModel 'conversation-list', view.content, @repository.call_center, @repository.user, @repository.conversation + view.start_ui = new z.ViewModel.StartUIViewModel 'start-ui', @repository.conversation, @repository.search, @repository.user, @repository.connect + view.archive = new z.ViewModel.ArchiveViewModel 'archive', @repository.conversation + view.actions = new z.ViewModel.ActionsViewModel 'actions-bubble', @repository.conversation, @repository.user, view.conversation_list + view.title = new z.ViewModel.WindowTitleViewModel view.content, @repository.user, @repository.conversation + view.welcome = new z.ViewModel.WelcomeViewModel 'welcome', @repository.user + view.settings = new z.ViewModel.SettingsViewModel 'self-settings', @repository.user, @repository.conversation, @repository.client, @repository.cryptography + view.warnings = new z.ViewModel.WarningsViewModel 'warnings' + view.modals = new z.ViewModel.ModalsViewModel 'modals' + + view.loading = new z.ViewModel.LoadingViewModel 'loading-screen', @repository.user + + return view + + _setup_utils: -> + utils = + debug: new z.util.DebugUtil @repository.user, @repository.conversation + return utils + + # Subscribe to amplify events. + _subscribe_to_events: -> + amplify.subscribe z.event.WebApp.SIGN_OUT, @logout + + + ############################################################################### + # Initialization + ############################################################################### + + ### + Initialize the app. + @note any failure will result in a logout + @todo Check if we really need to logout the user in all these error cases or how to recover from them + ### + init_app: (is_reload = @_is_reload()) => + @_load_access_token is_reload + .then => + @view.loading.switch_message z.string.init_received_access_token, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.RECEIVED_ACCESS_TOKEN + return z.util.protobuf.load_protos "ext/proto/generic-message-proto/messages.proto?#{z.util.Environment.version false}" + .then => + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.INITIALIZED_PROTO_MESSAGES + return @_get_user_self() + .then (self_user_et) => + @view.loading.switch_message z.string.init_received_self_user, true + @repository.client.init self_user_et + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.RECEIVED_SELF_USER + return @repository.storage.init false + .then => + @view.loading.switch_message z.string.init_initialized_storage, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.INITIALIZED_STORAGE + number_of_sessions = Object.keys(@repository.storage.sessions).length + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.SESSIONS, number_of_sessions, 50 + return @repository.cryptography.init() + .then => + @view.loading.switch_message z.string.init_initialized_cryptography, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.INITIALIZED_CRYPTOGRAPHY + return @repository.client.get_valid_local_client() + .then (client_observable) => + @view.loading.switch_message z.string.init_validated_client, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.VALIDATED_CLIENT + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.CLIENT_TYPE, client_observable().type + @repository.cryptography.current_client = client_observable + @repository.event.current_client = client_observable + @repository.event.connect() + promises = [ + @repository.client.get_clients_for_self() + @repository.user.init_properties() + @repository.conversation.get_conversations() + @repository.user.get_connections() + ] + return Promise.all promises + .then (response_array) => + [client_ets, user_properties, conversation_ets, connection_ets] = response_array + @view.loading.switch_message z.string.init_received_user_data, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.RECEIVED_USER_DATA + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.CLIENTS, client_ets.length + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.CONVERSATIONS, conversation_ets.length, 50 + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.CONNECTIONS, connection_ets.length, 50 + @repository.user.self().devices client_ets + + @logger.log @logger.levels.INFO, 'Mapping user connections to conversations' + amplify.publish z.event.WebApp.CONVERSATION.MAP_CONNECTION, @repository.user.connections() + @_subscribe_to_beforeunload() + return @repository.event.update_from_notification_stream() + .then (notifications_count) => + @view.loading.switch_message z.string.init_updated_from_notifications, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.UPDATED_FROM_NOTIFICATIONS + @telemetry.add_statistic z.telemetry.app_init.AppInitStatisticsValue.NOTIFICATIONS, notifications_count, 100 + return @_watch_online_status() + .then => + @view.loading.switch_message z.string.init_app_pre_loaded, true + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.APP_PRE_LOADED + @logger.log @logger.levels.INFO, 'App pre-loading completed' + @_show_ui() + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.SHOWING_UI + @telemetry.report() + amplify.publish z.event.WebApp.LOADED + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.APP_LOADED + return @repository.conversation.update_conversations @repository.conversation.conversations_unarchived() + .then => + @repository.announce.init() + @telemetry.time_step z.telemetry.app_init.AppInitTimingsStep.UPDATED_CONVERSATIONS + @logger.log @logger.levels.INFO, 'App fully loaded' + .catch (error) => + @logger.log @logger.levels.INFO, 'Error during app initialization.' + @logger.log @logger.levels.DEBUG, + "App reload: '#{is_reload}', Document referrer: '#{document.referrer}', Location: '#{window.location.href}'" + error_message = error.message or error + invalid_client = [z.client.ClientError::TYPE.MISSING_ON_BACKEND, z.client.ClientError::TYPE.NO_LOCAL_CLIENT] + if is_reload and error.type not in invalid_client + @auth.client.execute_on_connectivity().then -> window.location.reload false + else if navigator.onLine + @logger.log @logger.levels.ERROR, + "Could not load app version '#{z.util.Environment.version false}': #{error_message}", error + @logout 'init_app' + else + @logger.log @logger.levels.WARN, 'No connectivity. Trigger reload on regained connectivity.', error + @_watch_online_status() + + ### + Get the self user from the backend. + @return [Promise] Promise that resolves with the self user entity + ### + _get_user_self: -> + return new Promise (resolve, reject) => + @repository.user.get_me() + .then (user_et) => + @logger.log @logger.levels.INFO, "Loaded self user with ID '#{user_et.id}'" + + if not user_et.email() and not user_et.phone() + reject new Error 'User does not have a verified identity' + else + @service.storage.init user_et.id + .then => + if not user_et.picture_medium().length + z.util.load_url_blob z.config.UNSPLASH_URL, (blob) => + @repository.user.change_picture blob, -> amplify.publish z.event.WebApp.WELCOME.UNSPLASH_LOADED + resolve user_et + .catch (error) => + error = new Error "Loading self user failed: #{error}" + @logger.log @logger.levels.ERROR, error.message + reject error + + ### + Check whether the page has been reloaded. + @private + @return [Boolean] True if it is a page refresh + ### + _is_reload: -> + is_reload = z.util.is_same_location document.referrer, window.location.href + @logger.log @logger.levels.DEBUG, + "App reload: '#{is_reload}', Document referrer: '#{document.referrer}', Location: '#{window.location.href}'" + return is_reload + + ### + Load the access token from cache or get one from the backend. + @return [Promise] Promise that resolves with the Access Token + ### + _load_access_token: (is_reload) -> + return new Promise (resolve) => + if z.util.Environment.frontend.is_localhost() or document.referrer.toLowerCase().includes '/auth' + token_promise = @auth.repository.get_cached_access_token().then(resolve) + else + token_promise = @auth.repository.get_access_token().then(resolve) + + token_promise.catch (error) => + if is_reload + if error.type is error.type is z.auth.AccessTokenError::TYPE.REQUEST_FORBIDDEN + @logger.log @logger.levels.ERROR, "Session expired on page reload: #{error.message}", error + Raygun.send new Error ('Session expired on page reload'), error + @_redirect_to_login true + else if error.type isnt z.auth.AccessTokenError::TYPE.NOT_FOUND_IN_CACHE + @logger.log @logger.levels.WARN, 'Connectivity issues. Trigger reload on regained connectivity.', error + @auth.client.execute_on_connectivity().then -> window.location.reload false + else if navigator.onLine + if error.type is z.auth.AccessTokenError::TYPE.NOT_FOUND_IN_CACHE + @logger.log @logger.levels.WARN, 'No access token found in cache. Redirecting to login.', error + @_redirect_to_login false + else if error.type is z.auth.AccessTokenError::TYPE.REQUEST_FORBIDDEN + @logger.log @logger.levels.WARN, 'Access token request forbidden. Redirecting to login.', error + @_redirect_to_login false + else + @logger.log @logger.levels.ERROR, "Could not get access token: #{error.message}. Logging out user.", error + @logout 'init_app' + else + @logger.log @logger.levels.WARN, 'No connectivity. Trigger reload on regained connectivity.', error + @_watch_online_status() + + # Subscribe to 'beforeunload to stop calls and disconnect the WebSocket. + _subscribe_to_beforeunload: -> + $(window).on 'beforeunload', => + @logger.log '\'window.onbeforeunload\' was triggered, so we will disconnect from the backend.' + @repository.event.disconnect z.event.WebSocketService::CHANGE_TRIGGER.PAGE_NAVIGATION + @repository.call_center.state_handler.leave_call_on_beforeunload() + @repository.storage.terminate() + return undefined + + # Hide the loading spinner and show the application UI. + _show_ui: -> + @logger.log @logger.levels.INFO, 'Showing application UI' + conversation_et = @repository.conversation.get_most_recent_conversation() + has_picture = !!@repository.user.self().picture_medium() + + connect_token = z.util.get_url_parameter z.auth.URLParameter.CONNECT + if connect_token + @logger.log @logger.levels.INFO, 'Found connect token' + @repository.user.create_connection_from_invite_token connect_token + + if conversation_et and has_picture + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + setTimeout => + types_to_notify = [z.conversation.ConversationType.REGULAR, z.conversation.ConversationType.ONE2ONE] + if conversation_et.type() in types_to_notify + @repository.system_notification.request_permission() + , 2000 + else if @repository.user.connect_requests().length > 0 and has_picture + setTimeout -> + amplify.publish z.event.WebApp.PENDING.SHOW + , 1000 + else if has_picture and not connect_token + amplify.publish z.event.WebApp.PROFILE.SHOW + setTimeout -> + amplify.publish z.event.WebApp.SEARCH.SHOW + , 1000 + else + # Triggered when newly registered user opens Wire (first run experience) + amplify.publish z.event.WebApp.PROFILE.SHOW, false + amplify.publish z.event.WebApp.APP.HIDE + setTimeout -> + amplify.publish z.event.WebApp.WELCOME.SHOW + , 1500 + + $('#loading-screen').remove() + $('#wire-main') + .removeClass 'off' + .attr 'data-uie-value', 'is-loaded' + + # Subscribe to 'navigator.onLine' related events. + _watch_online_status: -> + @logger.log @logger.levels.INFO, 'Watching internet connectivity status' + $(window).on 'offline', @on_internet_connection_lost + $(window).on 'online', @on_internet_connection_gained + + # Behavior when internet connection is re-established. + on_internet_connection_gained: => + @logger.log @logger.levels.INFO, 'Internet connection regained. Re-establishing WebSocket connection...' + @auth.client.execute_on_connectivity() + .then => + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.NO_INTERNET + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + @repository.event.reconnect z.event.WebSocketService::CHANGE_TRIGGER.ONLINE + + # Reflect internet connection loss in the UI. + on_internet_connection_lost: => + @logger.log @logger.levels.WARN, 'Internet connection lost' + @repository.event.disconnect z.event.WebSocketService::CHANGE_TRIGGER.OFFLINE + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.NO_INTERNET + + + ############################################################################### + # Logout + ############################################################################### + + ### + Logs the user out on the backend and deletes cached data. + @param cause [String] Cause for logout + @param clear_data [Boolean] Whether to keep data in database + @param session_expired [Boolean] Whether to redirect the user to the login page + ### + logout: (cause, clear_data = false, session_expired = false) => + _logout = => + # Disconnect from our backend, end tracking and clear cached data + @repository.event.disconnect z.event.WebSocketService::CHANGE_TRIGGER.LOGOUT + amplify.publish z.event.WebApp.ANALYTICS.SESSION.CLOSE + + # Clear Local Storage (but don't delete the cookie label if you were logged in with a permanent client) + do_not_delete = [z.storage.StorageKey.AUTH.SHOW_LOGIN] + + if @repository.client.is_current_client_permanent() and not clear_data + do_not_delete.push z.storage.StorageKey.AUTH.PERSIST + + # XXX remove on next iteration + self_user = @repository.user.self() + if self_user + cookie_label_key = @repository.client.construct_cookie_label_key self_user.email() or self_user.phone() + + amplify_objects = amplify.store() + for amplify_key, amplify_value of amplify_objects + continue if amplify_key is cookie_label_key and clear_data + do_not_delete.push amplify_key if z.util.contains amplify_key, z.storage.StorageKey.AUTH.COOKIE_LABEL + + @repository.cache.clear_cache session_expired, do_not_delete + + # Clear IndexedDB + if clear_data + @repository.storage.delete_everything() + .catch (error) => + @logger.log @logger.levels.ERROR, 'Failed to delete database before logout', error + .then => + @_redirect_to_login session_expired + else + @_redirect_to_login session_expired + + _logout_on_backend = => + @logger.log @logger.levels.INFO, "Logout triggered by '#{cause}': Disconnecting user from the backend." + @auth.repository.logout() + .then -> _logout() + .catch => @_redirect_to_login false + + if session_expired + _logout() + else if navigator.onLine + _logout_on_backend() + else + @logger.log @logger.levels.WARN, 'No internet access. Continuing when internet connectivity regained.' + $(window).on 'online', -> _logout_on_backend() + + # Redirect to the login page after internet connectivity has been verified. + _redirect_to_login: (session_expired) -> + @logger.log @logger.levels.INFO, + "Redirecting to login after connectivity verification. Session expired: #{session_expired}" + @auth.client.execute_on_connectivity() + .then -> + url = "/auth/#{location.search}" + url = z.util.append_url_parameter url, z.auth.URLParameter.EXPIRED if session_expired + window.location.replace url + + + ############################################################################### + # Debugging + ############################################################################### + + # Disable debugging on any environment. + disable_debugging: -> + amplify.publish z.event.WebApp.PROPERTIES.CHANGE.DEBUG, false + + # Enable debugging on any environment. + enable_debugging: -> + amplify.publish z.event.WebApp.PROPERTIES.CHANGE.DEBUG, true + + # Report call telemetry to Raygun for analysis. + report_call: => + @repository.call_center.report_call() + + # Reset all known sessions at once. + reset_all_sessions: => + @repository.conversation.reset_all_sessions() + + # Initialize debugging features. + init_debugging: => + return if not z.ViewModel.DebugViewModel + @_attach_live_reload() if z.util.Environment.frontend.is_localhost() + @_initialize_sidebar() if not z.util.Environment.frontend.is_production() + + # Attach live reload on localhost. + _attach_live_reload: -> + live_reload = document.createElement 'script' + live_reload.id = 'live_reload' + live_reload.src = 'http://localhost:32123/livereload.js' + document.body.appendChild live_reload + $('html').addClass 'development' + + # Initialize the view model for the debug sidebar. + _initialize_sidebar: -> + @view.debug_view = new z.ViewModel.DebugViewModel 'debug', @repository.conversation, @repository.user + + +############################################################################### +# Setting up the App +############################################################################### +$ -> + if $('#wire-main-app').length isnt 0 + wire.app = new z.main.App wire.auth diff --git a/app/script/main/auth.coffee b/app/script/main/auth.coffee new file mode 100644 index 00000000000..103ad203b2a --- /dev/null +++ b/app/script/main/auth.coffee @@ -0,0 +1,36 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.main ?= {} + +class z.main.Auth + ### + Constructs objects needed for app authentication. + + @param [Object] settings Collection of URL settings + @option settings [String] environment Handle of the backend environment (staging, edge, etc.) + @option settings [String] web_socket_url URL to the backend's WebSocket + @option settings [String] rest_url URL to the backend's REST service + @option settings [String] parameter Additional parameters for the webapp's login URL + ### + constructor: (@settings) -> + @audio = new z.audio.AudioRepository() + @client = new z.service.Client @settings + @service = new z.auth.AuthService @client + @repository = new z.auth.AuthRepository @service diff --git a/app/script/main/auth_init_dist.coffee b/app/script/main/auth_init_dist.coffee new file mode 100644 index 00000000000..cb8927c32bc --- /dev/null +++ b/app/script/main/auth_init_dist.coffee @@ -0,0 +1,43 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +############################################################################### +# Setting up the Environment (DIST) +############################################################################### +$ -> + env = z.util.get_url_parameter z.auth.URLParameter.ENVIRONMENT + if env? and env is z.service.BackendEnvironment.EDGE + settings = + environment: z.service.BackendEnvironment.EDGE + rest_url: 'https://edge-nginz-https.zinfra.io' + web_socket_url: 'wss://edge-nginz-ssl.zinfra.io' + settings.parameter = '?env=edge' if env? + else if env? and env is z.service.BackendEnvironment.STAGING + settings = + environment: z.service.BackendEnvironment.STAGING + rest_url: 'https://staging-nginz-https.zinfra.io' + web_socket_url: 'wss://staging-nginz-ssl.zinfra.io' + settings.parameter = '?env=staging' if env? + else + settings = + environment: z.service.BackendEnvironment.PRODUCTION + rest_url: 'https://prod-nginz-https.wire.com' + web_socket_url: 'wss://prod-nginz-ssl.wire.com' + + window.wire = + auth: new z.main.Auth settings diff --git a/app/script/main/auth_init_edge.coffee b/app/script/main/auth_init_edge.coffee new file mode 100644 index 00000000000..dc624c39218 --- /dev/null +++ b/app/script/main/auth_init_edge.coffee @@ -0,0 +1,43 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +############################################################################### +# Setting up the Environment (EDGE) +############################################################################### +$ -> + env = z.util.get_url_parameter z.auth.URLParameter.ENVIRONMENT + if env? and env is 'prod' + settings = + environment: z.service.BackendEnvironment.PRODUCTION + rest_url: 'https://prod-nginz-https.wire.com' + web_socket_url: 'wss://prod-nginz-ssl.wire.com' + settings.parameter = '?env=prod' if env? + else if env? and env is 'staging' + settings = + environment: z.service.BackendEnvironment.STAGING + rest_url: 'https://staging-nginz-https.zinfra.io' + web_socket_url: 'wss://staging-nginz-ssl.zinfra.io' + settings.parameter = '?env=staging' if env? + else + settings = + environment: z.service.BackendEnvironment.EDGE + rest_url: 'https://edge-nginz-https.zinfra.io' + web_socket_url: 'wss://edge-nginz-ssl.zinfra.io' + + window.wire = + auth: new z.main.Auth settings diff --git a/app/script/main/auth_init_prod.coffee b/app/script/main/auth_init_prod.coffee new file mode 100644 index 00000000000..983c77d7346 --- /dev/null +++ b/app/script/main/auth_init_prod.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +############################################################################### +# Setting up the Environment (PRODUCTION) +############################################################################### +$ -> + settings = + environment: z.service.BackendEnvironment.PRODUCTION + rest_url: 'https://prod-nginz-https.wire.com' + web_socket_url: 'wss://prod-nginz-ssl.wire.com' + + window.wire = + auth: new z.main.Auth settings diff --git a/app/script/main/auth_init_staging.coffee b/app/script/main/auth_init_staging.coffee new file mode 100644 index 00000000000..14f2e5c3e30 --- /dev/null +++ b/app/script/main/auth_init_staging.coffee @@ -0,0 +1,43 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +############################################################################### +# Setting up the Environment (STAGING) +############################################################################### +$ -> + env = z.util.get_url_parameter z.auth.URLParameter.ENVIRONMENT + if env? and env is 'prod' + settings = + environment: z.service.BackendEnvironment.PRODUCTION + rest_url: 'https://prod-nginz-https.wire.com' + web_socket_url: 'wss://prod-nginz-ssl.wire.com' + settings.parameter = '?env=prod' if env? + else if env? and env is z.service.BackendEnvironment.EDGE + settings = + environment: z.service.BackendEnvironment.EDGE + rest_url: 'https://edge-nginz-https.zinfra.io' + web_socket_url: 'wss://edge-nginz-ssl.zinfra.io' + settings.parameter = '?env=edge' if env? + else + settings = + environment: z.service.BackendEnvironment.STAGING + rest_url: 'https://staging-nginz-https.zinfra.io' + web_socket_url: 'wss://staging-nginz-ssl.zinfra.io' + + window.wire = + auth: new z.main.Auth settings diff --git a/app/script/media/MediaEmbeds.coffee b/app/script/media/MediaEmbeds.coffee new file mode 100644 index 00000000000..b8e2093c095 --- /dev/null +++ b/app/script/media/MediaEmbeds.coffee @@ -0,0 +1,236 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.media ?= {} + +# Media embeds. +z.media.MediaEmbeds = do -> + + ### + Create and iframe. + + @private + + @param options [Object] Settings to be used to create the iframe + ### + _create_iframe_container = (options) -> + + defaults = + allowfullscreen: ' allowfullscreen' + class: 'iframe-container iframe-container-video' + frameborder: '0' + height: '100%' + type: 'default' + video: true + width: '100%' + + opt = _.extend defaults, options + + iframe_container = '
    ' + + if not opt.video + opt.allowfullscreen = '' + opt.class = 'iframe-container' + + if z.util.Environment.electron + opt.allowfullscreen = '' + + return z.util.string_format iframe_container, opt.class, opt.width, opt.height, opt.src, opt.frameborder, opt.allowfullscreen + + # Enum of different regex for the supported services. + _regex = + # example: http://regexr.com/3ase5 + youtube: /.*(?:youtu.be|youtube.com).*(?:\/|v\/|u\/\w\/|embed\/|watch\?v=)([^#\&\?]*).*/g + soundcloud: /(https?:\/\/(?:www\.|m\.)?)?soundcloud\.com(\/[\w\-]+){2,3}/g + spotify: /https?:\/\/(?:play\.|open\.)*spotify.com\/([\w\-/]+)/g + vimeo: /https?:\/\/(?:vimeo\.com\/|player\.vimeo\.com\/)(?:video\/|(?:channels\/staffpicks\/|channels\/)|)((\w|-){7,9})/g + + ### + Appends and iFrame. + + @private + + @param link [HTMLAnchorElement] + @param message [String] Message containing the link + @param iframe [String] HTML of iframe + ### + _append_iframe = (link, message, iframe) -> + link_string = link.outerHTML.replace /&/g, '&' + message = message.replace link_string, "#{link_string}#{iframe}" + + ### + Generate embed url to use as src in iframes + + @private + + @param url [String] given youtube url + ### + _generate_youtube_embed_url = (url) -> + if url.match /youtu.be|youtube.com/ + + # get youtube video id + video_id = url.match /(?:embed\/|v=|v\/|be\/)([a-zA-Z0-9_-]{11})/ + + return if not video_id + + # we have to remove the v param and convert the timestamp + # into an embed friendly format (start=seconds) + query = url + .substr url.indexOf('?'), url.length + .replace /^[?]/, '&' + .replace /[&]v=[a-zA-Z0-9_-]{11}/, '' + .replace /[&#]t=([a-z0-9]+)/, (a, b) -> "&start=#{_convert_youtube_timestamp_to_seconds b}" + .replace /[&]?autoplay=1/, '' # remove autoplay param + + # append html5 parameter to youtube src to force html5 mode + # this fixes the issue that FF displays black box in some cases + return "https://www.youtube.com/embed/#{video_id[1]}?html5=1#{query}" + + ### + Converts youtube timestamp into seconds + + @private + + @param timestamp [String] youtube timestamp 1h8m55s + ### + _convert_youtube_timestamp_to_seconds = (timestamp) -> + return 0 if not timestamp + return window.parseInt(timestamp, 10) if /^[0-9]*$/.test timestamp + hours = timestamp.match(/([0-9]*)h/)?[1] or 0 + minutes = timestamp.match(/([0-9]*)m/)?[1] or 0 + seconds = timestamp.match(/([0-9]*)s/)?[1] or 0 + return window.parseInt(hours, 10) * 3600 + window.parseInt(minutes, 10) * 60 + window.parseInt(seconds, 10) + + # Make public for testability. + regex: _regex + generate_youtube_embed_url: _generate_youtube_embed_url + convert_youtube_timestamp_to_seconds: _convert_youtube_timestamp_to_seconds + + ### + Appends SoundCloud iFrame if link is a valid SoundCloud source. + + @param link [HTMLAnchorElement] + @param message [String] Message containing the link + ### + soundcloud: (link, message) -> + link_src = link.href + + if link_src.match _regex.soundcloud + link_src = link_src.replace /(m\.)/, '' + link_path_name = link.pathname + + if link_path_name.endsWith('/') + link_path_name = link_path_name.substr 0, link_path_name.length - 1 + + is_single_track = false + slashes_in_link = link_path_name.split('/').length + + if slashes_in_link is 3 + is_single_track = true + else if slashes_in_link > 3 and link_path_name.indexOf('sets') is -1 + # Fix for WEBAPP-1137 + return message + + height = if is_single_track then 164 else 465 + + iframe = _create_iframe_container + src: 'https://w.soundcloud.com/player/?url={1}&visual=false&show_comments=false&buying=false&show_playcount=false&liking=false&sharing=false&hide_related=true' + type: 'soundcloud' + video: false + height: height + + embed = z.util.string_format iframe, height, link_src + message = _append_iframe link, message, embed + + return message + + ### + Appends Spotify iFrame if link is a valid Spotify source. + + @param link [HTMLAnchorElement] + @param message [String] Message containing the link + ### + spotify: (link, message) -> + link_src = link.href + + if link_src.match _regex.spotify + + iframe = _create_iframe_container + src: 'https://embed.spotify.com/?uri=spotify$1' + type: 'spotify' + video: false + height: '80px' + + # convert spotify uri: album/23... -> album:23... -> album%3A23... + embed = '' + + link_src.replace _regex.spotify, (match, group1) -> + replace_slashes = group1.replace /\//g, ':' + encoded_params = encodeURIComponent ":#{replace_slashes}" + embed = iframe.replace '$1', encoded_params + + message = _append_iframe link, message, embed + + return message + + ### + Appends Vimeo iFrame if link is a valid Vimeo source. + + @param link [HTMLAnchorElement] + @param message [String] Message containing the link + ### + vimeo: (link, message, theme_color) -> + link_src = link.href + vimeo_color = theme_color?.replace '#', '' + + if link_src.match _regex.vimeo + return message if z.util.contains link_src, '/user' + + iframe = _create_iframe_container + src: "https://player.vimeo.com/video/$1?portrait=0&color=#{vimeo_color}&badge=0" + type: 'vimeo' + + embed = '' + + link_src.replace _regex.vimeo, (match, group1) -> + embed = iframe.replace '$1', group1 + + message = _append_iframe link, message, embed + + return message + + ### + Appends YouTube iFrame if link is a valid YouTube source. + + @param link [HTMLAnchorElement] + @param message [String] Message containing the link + ### + youtube: (link, message) -> + embed_url = _generate_youtube_embed_url link.href + + if embed_url + + iframe = _create_iframe_container + src: embed_url + type: 'youtube' + + message = _append_iframe link, message, iframe + return message + + return message diff --git a/app/script/media/MediaParser.coffee b/app/script/media/MediaParser.coffee new file mode 100644 index 00000000000..b4dd7fd17a4 --- /dev/null +++ b/app/script/media/MediaParser.coffee @@ -0,0 +1,52 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.media ?= {} + +# Media Parser to render rich media embeds. +class MediaParser + # Construct a new Media parser and select available embeds. + constructor: -> + @embeds = [ + z.media.MediaEmbeds.soundcloud + z.media.MediaEmbeds.spotify + z.media.MediaEmbeds.vimeo + z.media.MediaEmbeds.youtube + ] + + ### + Render media embeds. + + @note Checks message for valid media links and appends an iFrame right after the link + + @param message [String] Message text + @param theme_color [String] Accent color to be applied to the embed + ### + render_media_embeds: (message, theme_color) => + div = document.createElement 'div' + div.innerHTML = message + links = div.querySelectorAll 'a' + + for link in links + for embed in @embeds + message = embed link, message, theme_color + + return message + +z.media.MediaParser = new MediaParser() diff --git a/app/script/message/CallMessageType.coffee b/app/script/message/CallMessageType.coffee new file mode 100644 index 00000000000..94ce62300af --- /dev/null +++ b/app/script/message/CallMessageType.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.message ?= {} + +# Enum for different call message types. +z.message.CallMessageType = + ACTIVATED: 'activated' + DEACTIVATED: 'deactivated' diff --git a/app/script/message/PingMessageType.coffee b/app/script/message/PingMessageType.coffee new file mode 100644 index 00000000000..b9044e2ec2b --- /dev/null +++ b/app/script/message/PingMessageType.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.message ?= {} + +# Enum for different ping message types. +z.message.PingMessageType = + PING: 'ping' + HOT_PING: 'hot-ping' diff --git a/app/script/message/SuperType.coffee b/app/script/message/SuperType.coffee new file mode 100644 index 00000000000..09a1e74a37a --- /dev/null +++ b/app/script/message/SuperType.coffee @@ -0,0 +1,36 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.message ?= {} + +### +Enum for different message super types. +### +z.message.SuperType = + ALL_VERIFIED: 'all-verified' + CALL: 'call' + CONVERSATION_RENAME: 'rename' # TODO used? + CONTENT: 'normal' + DEVICE: 'device' + LOCATION: 'location' + MEMBER: 'member' + PING: 'ping' + SPECIAL: 'special' + SYSTEM: 'system' + UNABLE_TO_DECRYPT: 'unable-to-decrypt' diff --git a/app/script/message/SystemMessageType.coffee b/app/script/message/SystemMessageType.coffee new file mode 100644 index 00000000000..e937ef7ae45 --- /dev/null +++ b/app/script/message/SystemMessageType.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.message ?= {} + +### +Enum for different system message types. + +@todo Refactor to use member-join and member-leave instead of normal. It duplicates "z.message.SuperType". +### +z.message.SystemMessageType = + NORMAL: 'normal' + MEMBER_JOIN: 'join' + MEMBER_LEAVE: 'leave' + CONVERSATION_CREATE: 'created-group' + CONVERSATION_RENAME: 'rename' + CONVERSATION_RESUME: 'resume' + CONNECTION_REQUEST: 'connecting' + CONNECTION_ACCEPTED: 'created-one-to-one' diff --git a/app/script/search/SearchLevel.coffee b/app/script/search/SearchLevel.coffee new file mode 100644 index 00000000000..435d412bcb2 --- /dev/null +++ b/app/script/search/SearchLevel.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.search ?= {} + +### +Enum of different search levels. +### +z.search.SEARCH_LEVEL = + DIRECT_CONTACT: 1 + FRIEND_OF_FRIEND: 2 + INDIRECT_CONTACT: 3 diff --git a/app/script/search/SearchMode.coffee b/app/script/search/SearchMode.coffee new file mode 100644 index 00000000000..188700eed2f --- /dev/null +++ b/app/script/search/SearchMode.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.search ?= {} + +# Enum of different search modes. +z.search.SEARCH_MODE = + COMMON_CONNECTIONS: 'common_connections' + CONTACTS: 'contacts' + SUGGESTIONS: 'suggestions' + TOP_PEOPLE: 'top_people' + ONBOARDING: 'onboarding' diff --git a/app/script/search/SearchRepository.coffee b/app/script/search/SearchRepository.coffee new file mode 100644 index 00000000000..ae0f157edd0 --- /dev/null +++ b/app/script/search/SearchRepository.coffee @@ -0,0 +1,221 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.search ?= {} + +# Search repository for all interactions with the search service. +class z.search.SearchRepository + ### + Construct a new Conversation Repository. + @param search_service [z.search.SearchService] Backend REST API search service implementation + @param user_repository [z.user.UserRepository] Repository for all user and connection interactions + ### + constructor: (@search_service, @user_repository) -> + @logger = new z.util.Logger 'z.search.SearchRepository', z.config.LOGGER.OPTIONS + + @search_result_mapper = new z.search.SearchResultMapper @user_repository + + @suggested_search_ets = [] + + amplify.subscribe z.event.WebApp.SEARCH.ONBOARDING, @show_onboarding + + ### + Get common contacts for given user. + @param user_id [String] User ID + @return [Promise>] Promise that will resolve with an array containing the users common contacts + ### + get_common_contacts: (user_id) => + @search_service.get_common user_id + .then (response) => + return @search_result_mapper.map_results response.documents, z.search.SEARCH_MODE.COMMON_CONNECTIONS + .then ([search_ets, mode]) => + return @_prepare_search_result search_ets, mode + + ### + Get "People you may know" (PYMK). + @param size [Integer] Number of requested user + @return [Promise] Promise that resolves with suggestions + ### + get_suggestions: -> + return new Promise (resolve, reject) => + @suggested_search_ets = z.storage.get_value(z.storage.StorageKey.SEARCH.SUGGESTED_SEARCH_ETS) or [] + + if @suggested_search_ets.length > 0 + @_prepare_search_result @suggested_search_ets, z.search.SEARCH_MODE.SUGGESTIONS + .then (user_ets) -> resolve user_ets + .catch (error) -> reject error + else + @search_service.get_suggestions z.config.SUGGESTIONS_FETCH_LIMIT + .then (response) => + return @search_result_mapper.map_results response.documents, z.search.SEARCH_MODE.SUGGESTIONS + .then ([search_ets, mode]) => + return @_prepare_search_result search_ets, mode + .then (search_ets) => + # store suggested user ids + @suggested_search_ets = search_ets + z.storage.set_value z.storage.StorageKey.SEARCH.SUGGESTED_SEARCH_ETS, @suggested_search_ets, 15 * 60 + resolve search_ets + .catch (error) -> + reject error + + ### + Get top people. + @return [Function] Promise that resolves with the top connections + ### + get_top_people: -> + @search_service.get_top z.config.TOP_PEOPLE_FETCH_LIMIT + .then (response) => + return @search_result_mapper.map_results response.documents, z.search.SEARCH_MODE.TOP_PEOPLE + .then ([search_ets, mode]) => + return @_prepare_search_result search_ets, mode + + ### + Ignore suggested user. + @param user_id [String] User ID + @return [Promise] Promise that resolves when a suggestion has been ignored + ### + ignore_suggestion: (user_id) -> + @search_service.put_suggestions_ignore user_id + .then => + search_et = ko.utils.arrayFirst @suggested_search_ets, (search_et) -> + search_et.id is user_id + + # remove ignored user from the suggested user ids + user_id_index = @suggested_search_ets.indexOf search_et + @suggested_search_ets.splice user_id_index, 1 if user_id_index > -1 + z.storage.set_value z.storage.StorageKey.SEARCH.SUGGESTED_SEARCH_ETS, @suggested_search_ets, 30 + + ### + Search for users on the backend by name. + @param name [String] Search query + @return [Promise] Promise that resolves with the search results + ### + search_by_name: (name) -> + @search_service.get_contacts name, 30, z.search.SEARCH_LEVEL.INDIRECT_CONTACT, 1 + .then (response) => + return @search_result_mapper.map_results response?.documents, z.search.SEARCH_MODE.CONTACTS + .then ([search_ets, mode]) => + return @_prepare_search_result search_ets, mode + + ### + Show on-boarding results. + @param response [Object] Server response + @return [Promise] Promise that resolves with the connections found through on-boarding + ### + show_onboarding: (response) => + @search_result_mapper.map_results response.results, z.search.SEARCH_MODE.ONBOARDING + .then ([search_ets, mode]) => + return @_prepare_search_result search_ets, mode + + ### + Preparing the search results for display. + + @note We skip a few results as connection changes need a while to reflect on the graph. + @private + + @param search_ets [Array] An array of mapped search result entities + @param mode [z.search.SEARCH_MODE] Search mode + @return [Promise] Promise that will resolve with search results + ### + _prepare_search_result: (search_ets, mode) -> + return new Promise (resolve, reject) => + user_ids = [] + user_ids.push user_et.id for user_et in search_ets + + @user_repository.get_users_by_id user_ids, (user_ets) => + result_user_ets = [] + for user_et in user_ets + ### + Skipping some results to adjust for slow graph updates. + + Only show connected people among your top people. + Do not show already connected people when uploading address book. + Only show unknown or cancelled people in suggestions. + ### + switch mode + when z.search.SEARCH_MODE.COMMON_CONNECTIONS + result_user_ets.push user_et + when z.search.SEARCH_MODE.CONTACTS + if not user_et.connected() + user_et.connection_level z.user.ConnectionLevel.NO_CONNECTION + result_user_ets.push user_et + when z.search.SEARCH_MODE.ONBOARDING + result_user_ets.push user_et if not user_et.connected() + when z.search.SEARCH_MODE.SUGGESTIONS + states_to_suggest = [z.user.ConnectionStatus.UNKNOWN, z.user.ConnectionStatus.CANCELLED] + result_user_ets.push user_et if user_et.connection().status() in states_to_suggest + when z.search.SEARCH_MODE.TOP_PEOPLE + result_user_ets.push user_et if user_et.connected() + else + result_user_ets.push user_et + + if mode in [z.search.SEARCH_MODE.CONTACTS, z.search.SEARCH_MODE.SUGGESTIONS] + @_add_mutual_friends(result_user_ets, search_ets).then(resolve).catch(reject) + else + resolve result_user_ets + + ### + Adding mutual friend entities to the search results. + + @private + + @param user_ets [Array] User entities + @param search_ets [Array] Search entities returned from the server + @return [Promise] Promise that resolves with search results that where mutual friends are added + ### + _add_mutual_friends: (user_ets, search_ets) -> + return new Promise (resolve) => + mutual_friend_ids = _.flatten(search_et.mutual_friend_ids for search_et in search_ets when search_et.mutual_friend_ids?) + + @user_repository.get_users_by_id mutual_friend_ids, (all_mutual_friend_ets) -> + for user_et in user_ets + search_et = ko.utils.arrayFirst search_ets, (search_et) -> + search_et.id is user_et.id + user_et.mutual_friends_total search_et.mutual_friends_total + if user_et.mutual_friends_total() > 0 + user_et.mutual_friend_ids search_et.mutual_friend_ids + mutual_friend_ets = [] + for mutual_friend_id in user_et.mutual_friend_ids() + mutual_friend_et = ko.utils.arrayFirst all_mutual_friend_ets, (user_et) -> + user_et.id is mutual_friend_id + mutual_friend_ets.push mutual_friend_et if mutual_friend_et? + user_et.mutual_friend_ets mutual_friend_ets + + if user_et.mutual_friend_ets().length is 1 + user_et.relation_info = z.localization.Localizer.get_text { + id: z.string.search_suggestion_one + replace: {placeholder: '%@.first_name', content: user_et.mutual_friend_ets()[0].first_name()} + } + else if user_et.mutual_friend_ets().length is 2 + user_et.relation_info = z.localization.Localizer.get_text { + id: z.string.search_suggestion_two + replace: [ + {placeholder: '%@.first_name', content: user_et.mutual_friend_ets()[0].first_name()} + {placeholder: '%@.other_name', content: user_et.mutual_friend_ets()[1].first_name()} + ] + } + else if user_et.mutual_friend_ets().length > 2 + user_et.relation_info = z.localization.Localizer.get_text { + id: z.string.search_suggestion_many + replace: [ + {placeholder: '%@.first_name', content: user_et.mutual_friend_ets()[0].first_name()} + {placeholder: '%no', content: user_et.mutual_friends_total() - 1} + ] + } + resolve user_ets diff --git a/app/script/search/SearchResultMapper.coffee b/app/script/search/SearchResultMapper.coffee new file mode 100644 index 00000000000..08fe2aed9c0 --- /dev/null +++ b/app/script/search/SearchResultMapper.coffee @@ -0,0 +1,46 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.search ?= {} + +# Search mapper to convert all server side JSON search results into usable entities. +class z.search.SearchResultMapper + # Construct a new Search Result Mapper. + constructor: -> + @logger = new z.util.Logger 'z.search.SearchResultMapper', z.config.LOGGER.OPTIONS + + ### + Converts JSON search results from other search modes into core entities. + + @param search_results [JSON] Search result data + @param mode [z.search.SEARCH_MODE] Search mode to be mapped + @return [Promise] Promise that will resolve with the mapped search results + ### + map_results: (search_results = [], mode) -> + return new Promise (resolve) -> + search_ets = [] + + for search_result in search_results + search_et = {} + search_et.id = search_result.id + if mode in [z.search.SEARCH_MODE.CONTACTS, z.search.SEARCH_MODE.SUGGESTIONS] + search_et.mutual_friends_total = search_result.total_mutual_friends + search_et.mutual_friend_ids = search_result.mutual_friends + search_ets.push search_et + resolve [search_ets, mode] diff --git a/app/script/search/SearchService.coffee b/app/script/search/SearchService.coffee new file mode 100644 index 00000000000..1b8f229bf21 --- /dev/null +++ b/app/script/search/SearchService.coffee @@ -0,0 +1,83 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.search ?= {} + +# Search service for all search calls to the backend REST API. +class z.search.SearchService + ### + Construct a new Search Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.search.SearchService', z.config.LOGGER.OPTIONS + + ### + Get common contacts for given user. + @param user_id [String] User ID + @return [Promise] Promise that resolves with the common contacts + ### + get_common: (user_id) -> + @client.send_request + type: 'GET' + url: @client.create_url "/search/common/#{user_id}" + + ### + Search for a user. + + @param query [String] Query string (case insensitive) + @param size [Integer] Number of requested user + @param level [Integer] How deep to search (friends, friends of friends, friends of friends or friends)... + @param directory [Integer] Fall back to directory if graph search does not find the size of people (0 or 1) + @return [Promise] Promise that resolves with the search results + ### + get_contacts: (query, size, level, directory) -> + @client.send_request + type: 'GET' + url: @client.create_url "/search/contacts?q=#{encodeURIComponent(query)}&size=#{size}&l=#{level}&d=#{directory}" + + ### + Get "People you may know" (PYMK). + @param size [Integer] Number of requested user + @return [Promise] Promise that resolves with the contact suggestions + ### + get_suggestions: (size) -> + @client.send_request + type: 'GET' + url: @client.create_url "/search/suggestions?size=#{size}" + + ### + Get top people. + @param [Integer] size number of requested user + @return [Promise] Promise that resolves with the top connections + ### + get_top: (size) -> + @client.send_request + type: 'GET' + url: @client.create_url "/search/top?size=#{size}" + + ### + Ignore suggested user. + @param user_id [String] User ID + @return [Promise] Promise that resolves when a suggestion has been ignored + ### + put_suggestions_ignore: (user_id) -> + @client.send_request + type: 'PUT' + url: @client.create_url "/search/suggestions/#{user_id}/ignore" diff --git a/app/script/service/BackendClientError.coffee b/app/script/service/BackendClientError.coffee new file mode 100644 index 00000000000..0b8cd8cb845 --- /dev/null +++ b/app/script/service/BackendClientError.coffee @@ -0,0 +1,78 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.service ?= {} + +class z.service.BackendClientError + constructor: (params) -> + @name = @constructor.name + @stack = (new Error()).stack + + if _.isObject params + @code = params.code + @label = params.label + @message = params.message + else if _.isNumber params + @code = params + @message = "#{params}" + + @:: = new Error() + @::constructor = @ + @::STATUS_CODE = + ACCEPTED: 202 + BAD_GATEWAY: 502 + BAD_REQUEST: 400 + CONFLICT: 409 + CONNECTIVITY_PROBLEM: 0 + CREATED: 201 + FORBIDDEN: 403 + INTERNAL_SERVER_ERROR: 500 + NO_CONTENT: 204 + NOT_FOUND: 404 + OK: 200 + REQUEST_TIMEOUT: 408 + PRECONDITION_FAILED: 412 + TOO_MANY_REQUESTS: 429 + UNAUTHORIZED: 401 + REQUEST_TOO_LARGE: 413 + + @::LABEL = + BAD_REQUEST: 'bad-request' + BLACKLISTED_EMAIL: 'blacklisted-email' + BLACKLISTED_PHONE: 'blacklisted-phone' + CONNECTIVITY_PROBLEM: 'connectivity-problem' + CONVERSATION_TOO_BIG: 'conv-too-big' + IN_USE: 'in-use' + INVALID_CREDENTIALS: 'invalid-credentials' + INVALID_EMAIL: 'invalid-email' + INVALID_INVITATION_CODE: 'invalid-invitation-code' + INVALID_OPERATION: 'invalid-op' + INVALID_PHONE: 'invalid-phone' + KEY_EXISTS: 'key-exists' + MISSING_AUTH: 'missing-auth' + MISSING_IDENTITY: 'missing-identity' + NOT_FOUND: 'not-found' + PASSWORD_EXISTS: 'password-exists' + PENDING_ACTIVATION: 'pending-activation' + PENDING_LOGIN: 'pending-login' + TOO_MANY_CLIENTS: 'too-many-clients' + TOO_MANY_MEMBERS: 'too-many-members' + UNAUTHORIZED: 'unauthorized' + UNKNOWN_CLIENT: 'unknown-client' + VOICE_CHANNEL_FULL: 'voice-channel-full' diff --git a/app/script/service/BackendEnvironment.coffee b/app/script/service/BackendEnvironment.coffee new file mode 100644 index 00000000000..f5b7437233f --- /dev/null +++ b/app/script/service/BackendEnvironment.coffee @@ -0,0 +1,25 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.service ?= {} + +z.service.BackendEnvironment = + EDGE: 'edge' + PRODUCTION: 'production' + STAGING: 'staging' diff --git a/app/script/service/Client.coffee b/app/script/service/Client.coffee new file mode 100644 index 00000000000..ef7045d270c --- /dev/null +++ b/app/script/service/Client.coffee @@ -0,0 +1,242 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.service ?= {} + +IGNORED_BACKEND_ERRORS = [ + z.service.BackendClientError::STATUS_CODE.BAD_GATEWAY + z.service.BackendClientError::STATUS_CODE.BAD_REQUEST + z.service.BackendClientError::STATUS_CODE.CONFLICT + z.service.BackendClientError::STATUS_CODE.CONNECTIVITY_PROBLEM + z.service.BackendClientError::STATUS_CODE.INTERNAL_SERVER_ERROR + z.service.BackendClientError::STATUS_CODE.NOT_FOUND + z.service.BackendClientError::STATUS_CODE.PRECONDITION_FAILED + z.service.BackendClientError::STATUS_CODE.REQUEST_TIMEOUT + z.service.BackendClientError::STATUS_CODE.TOO_MANY_REQUESTS +] + +# Client for all backend REST API calls. +class z.service.Client + ### + Construct a new client. + + @param settings [Object] Settings for different backend environments + @option settings [String] environment + @option settings [String] rest_url + @option settings [String] web_socket_url + @option settings [String] parameter + ### + constructor: (settings) -> + @logger = new z.util.Logger 'z.service.Client', z.config.LOGGER.OPTIONS + + z.util.Environment.backend.current = settings.environment + @rest_url = settings.rest_url + @web_socket_url = settings.web_socket_url + + @request_queue = [] + @is_requesting_access_token = ko.observable false + @is_requesting_access_token.subscribe (is_requesting) => + @execute_request_queue() if @access_token isnt '' and @request_queue.length > 0 and not is_requesting + + @access_token = '' + @access_token_type = '' + + @number_of_requests = ko.observable 0 + @number_of_requests.subscribe (new_value) -> + amplify.publish z.event.WebApp.TELEMETRY.BACKEND_REQUESTS, new_value + + # http://stackoverflow.com/a/18996758/451634 + pre_filters = $.Callbacks() + pre_filters.before_each_request = (options, originalOptions, jqXHR) => + jqXHR.wire = + original_request_options: originalOptions + request_id: @number_of_requests() + requested: new Date() + + $.ajaxPrefilter pre_filters.before_each_request + + ### + Create a request URL. + @param url [String] API endpoint to be prefixed with REST API environment + @return [String] REST API endpoint + ### + create_url: (url) -> + return "#{@rest_url}#{url}" + + ### + Request backend status. + @return [$.Promise] jquery ajax promise + ### + status: => + $.ajax + type: 'HEAD' + timeout: 500 + url: @create_url '/self' + + ### + Delay a function call until backend connectivity is guaranteed. + @return [Promise] Promise that is resolved when connectivity is verified + ### + execute_on_connectivity: => + return new Promise (resolve) => + check_status = => + @logger.log @logger.levels.INFO, 'Checking connectivity status' + @status() + .done => + @logger.log @logger.levels.INFO, 'Connectivity verified.' + resolve() + .fail (jqXHR) => + if jqXHR.readyState is 4 + @logger.log @logger.levels.INFO, "Connectivity verified by server error: #{jqXHR.statusText}" + resolve() + else + window.setTimeout check_status, 2000 + + check_status() + + # Execute queued requests. + execute_request_queue: => + @logger.log @logger.levels.INFO, "Executing '#{@request_queue.length}' queued requests" + for request in @request_queue + [config, request_resolve, request_reject] = @request_queue.shift() + @send_request config + .then (response) => + @logger.log @logger.levels.INFO, "Queued '#{config.type}' request to '#{config.url}' executed", response + request_resolve response + .catch (error) => + @logger.log @logger.levels.INFO, + "Failed to execute queued '#{config.type}' request to '#{config.url}'", error + request_reject error + + ### + Send jQuery AJAX request. + @see http://api.jquery.com/jquery.ajax/#jQuery-ajax-settings + @param config [jQuery.ajax SettingsObject] + ### + send_request: (config) -> + return new Promise (resolve, reject) => + if @is_requesting_access_token() + @logger.log @logger.levels.INFO, 'Request queued while access token is refreshed', config + @request_queue.push [config, resolve, reject] + else + headers = config.headers or {} + headers['Authorization'] = "#{@access_token_type} #{@access_token}" if @access_token + + xhrFields = {} + xhrFields['withCredentials'] = true if config.withCredentials + + @number_of_requests @number_of_requests() + 1 + + if _.isArray config.callback + $.ajax + contentType: config.contentType + data: config.data + headers: headers + processData: config.processData + timeout: config.timeout + type: config.type + url: config.url + xhrFields: xhrFields + .always (data_or_jqXHR, textStatus, jqXHR_or_data) => + if textStatus not in ['error', 'timeout'] + if jqXHR_or_data.wire + jqXHR_or_data.wire.original_request_options.api_endpoint = config.api_endpoint + jqXHR_or_data.wire.responded = new Date() + resolve [data_or_jqXHR, jqXHR_or_data] + else + switch data_or_jqXHR.status + when z.service.BackendClientError::STATUS_CODE.CONNECTIVITY_PROBLEM + @logger.log @logger.levels.WARN, 'Request failed due to connectivity problem.', config + @request_queue.push [config, resolve, reject] + @execute_on_connectivity().then => @execute_request_queue() + when z.service.BackendClientError::STATUS_CODE.UNAUTHORIZED + @logger.log @logger.levels.WARN, 'Request failed as access token is invalid.', config + @request_queue.push [config, resolve, reject] + amplify.publish z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEW + else + reject data_or_jqXHR.responseJSON or new z.service.BackendClientError data_or_jqXHR.status + else + $.ajax + contentType: config.contentType + data: config.data + headers: headers + processData: config.processData + timeout: config.timeout + type: config.type + url: config.url + .done (data, textStatus, jqXHR) => + @logger.log @logger.levels.OFF, "Server Response ##{jqXHR.wire.request_id} from '#{config.url}':", data + config.callback? data + resolve data + .fail (jqXHR, textStatus, errorThrown) => + switch jqXHR.status + when z.service.BackendClientError::STATUS_CODE.CONNECTIVITY_PROBLEM + @logger.log @logger.levels.WARN, 'Request failed due to connectivity problem.' + @request_queue.push [config, resolve, reject] + @execute_on_connectivity().then => @execute_request_queue() + return + when z.service.BackendClientError::STATUS_CODE.UNAUTHORIZED + @request_queue.push [config, resolve, reject] + @logger.log @logger.levels.WARN, 'Request failed as access token is invalid.' + amplify.publish z.event.WebApp.CONNECTION.ACCESS_TOKEN.RENEW + return + when z.service.BackendClientError::STATUS_CODE.FORBIDDEN + switch jqXHR.responseJSON?.label + when z.service.BackendClientError::LABEL.INVALID_CREDENTIALS + Raygun.send new Error 'Server request failed: Invalid credentials' + when z.service.BackendClientError::LABEL.TOO_MANY_CLIENTS, z.service.BackendClientError::LABEL.TOO_MANY_MEMBERS + @logger.log @logger.levels.WARN, "Server request failed: '#{jqXHR.responseJSON.label}'" + else + Raygun.send new Error 'Server request failed' + else + if jqXHR.status not in IGNORED_BACKEND_ERRORS + Raygun.send new Error "Server request failed: #{jqXHR.status}" + + if _.isFunction config.callback + config.callback null, jqXHR.responseJSON or new z.service.BackendClientError errorThrown + else + if navigator.onLine + reject jqXHR.responseJSON or new z.service.BackendClientError jqXHR.status + else + error_data = + code: z.service.BackendClientError::STATUS_CODE.CONNECTIVITY_PROBLEM + label: z.service.BackendClientError::LABEL.CONNECTIVITY_PROBLEM + message: 'Problem with the network connectivity' + reject new z.service.BackendClientError error_data + + ### + Send AJAX request with compressed JSON body. + + @param config [Object] + @option config [Function] callback + @option config [JSON] data + @option config [String] type + @option config [String] url + ### + send_json: (config) -> + @send_request + api_endpoint: config.api_endpoint + callback: config.callback + contentType: 'application/json; charset=utf-8' + data: pako.gzip JSON.stringify config.data if config.data + headers: + 'Content-Encoding': 'gzip' + processData: false + type: config.type + url: config.url diff --git a/app/script/storage/SkipError.coffee b/app/script/storage/SkipError.coffee new file mode 100644 index 00000000000..ce16d39ab67 --- /dev/null +++ b/app/script/storage/SkipError.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.storage ?= {} + +class z.storage.SkipError + constructor: (message) -> + @name = @constructor.name + @message = message + @stack = (new Error()).stack + + @:: = new Error() + @::constructor = @ diff --git a/app/script/storage/StorageKey.coffee b/app/script/storage/StorageKey.coffee new file mode 100644 index 00000000000..2618ba2af2e --- /dev/null +++ b/app/script/storage/StorageKey.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.storage ?= {} + +z.storage.StorageKey = + AUTH: + ACCESS_TOKEN: + VALUE: 'z.storage.StorageKey.AUTH.ACCESS_TOKEN.VALUE' + EXPIRATION: 'z.storage.StorageKey.AUTH.ACCESS_TOKEN.EXPIRATION' + TTL: 'z.storage.StorageKey.AUTH.ACCESS_TOKEN.TTL' + TYPE: 'z.storage.StorageKey.AUTH.ACCESS_TOKEN.TYPE' + COOKIE_LABEL: 'z.storage.StorageKey.AUTH.COOKIE_LABEL' + PERSIST: 'z.storage.StorageKey.AUTH.PERSIST' + SHOW_LOGIN: 'z.storage.StorageKey.AUTH.SHOW_LOGIN' + CONVERSATION: + INPUT: 'z.storage.StorageKey.CONVERSATION.INPUT' + LOCALIZATION: + LOCALE: 'z.storage.StorageKey.LOCALIZATION.LOCALE' + SEARCH: + SUGGESTED_SEARCH_ETS: 'z.storage.StorageKey.SEARCH.SUGGESTED_SEARCH_ETS' + ANNOUNCE: + ANNOUNCE_KEY: 'z.storage.StorageKey.ANNOUNCE.ANNOUNCE_KEY' diff --git a/app/script/storage/StorageRepository.coffee b/app/script/storage/StorageRepository.coffee new file mode 100644 index 00000000000..4e694c39cb1 --- /dev/null +++ b/app/script/storage/StorageRepository.coffee @@ -0,0 +1,427 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.storage ?= {} + +class z.storage.StorageRepository extends cryptobox.CryptoboxStore + constructor: (@storage_service) -> + @logger = new z.util.Logger 'z.storage.StorageRepository', z.config.LOGGER.OPTIONS + + @identity = undefined + @prekeys = {} + @sessions = {} + + @sessions_handled = 0 + @sessions_promises = [] + @sessions_total = 0 + @sessions_queue = ko.observableArray [] + + @sessions_queue.subscribe (sessions) => + if sessions.length > 0 + @_deserialize_session @sessions_queue()[0] + .then (session) => + if session + @logger.log @logger.levels.INFO, "De-serialized session '#{@sessions_queue()[0].id}'", session + @sessions_handled++ + if @sessions_handled % 5 is 0 + replace = [@sessions_handled, @sessions_total] + amplify.publish z.event.WebApp.APP.UPDATE_INIT, z.string.init_sessions_progress, false, replace + window.setTimeout => + @sessions_queue.shift() + , 0 + else + @logger.log @logger.levels.INFO, "De-serialized '#{Object.keys(@sessions).length}' sessions", @sessions + @sessions_promises[0] @sessions + + + ############################################################################### + # Initialization + ############################################################################### + + ### + Initialize the repository to setup Cryptobox with the Local Identity Key Pair, Sessions and Pre-Keys. + @return [Promise] Promise that will resolve with the repository after initialization + ### + init: (skip_sessions) => + return new Promise (resolve, reject) => + @_load_identity() + .then (@identity) => + if @identity + @logger.log @logger.levels.INFO, 'Loaded local identity key pair from database', @identity + else + @logger.log @logger.levels.INFO, 'We did not find a local identity. This is a new client.' + return @identity + .then (local_identity) => + return {} if not local_identity + if skip_sessions + throw new z.storage.SkipError 'Skipped loading of sessions and pre-keys' + else + return @_load_sessions() + .then => + return @_load_pre_keys() + .then => + @logger.log @logger.levels.INFO, 'Initialized repository' + resolve @ + .catch (error) => + if error instanceof z.storage.SkipError + @logger.log "Initialized repository with the following exception: #{error.message}" + resolve @ + else + @logger.log @logger.levels.ERROR, "Storage Repository initialization failed: #{error?.message}", error + reject error + + _deserialize_session: (session) -> + return Promise.resolve() + .then => + try + bytes = sodium.from_base64 session.serialised + @sessions[session.id] = Proteus.session.Session.deserialise @identity, bytes.buffer + return @sessions[session.id] + catch error + @logger.log @logger.levels.ERROR, "Session '#{session.id}' is corrupt.", error + # TODO: Consider deleting or repairing corrupt data (like broken sessions) + return undefined + + ### + Loads the user's identity key pair. + @private + @return [Promise] Promise that will resolve with the key pair loaded from storage + ### + _load_identity: => + return new Promise (resolve, reject) => + @storage_service.load @storage_service.OBJECT_STORE_KEYS, 'local_identity' + .then (identity_key_pair) -> + if identity_key_pair + bytes = sodium.from_base64 identity_key_pair.serialised + resolve Proteus.keys.IdentityKeyPair.deserialise bytes.buffer + else + resolve undefined + .catch (error) => + @logger.log @logger.levels.ERROR, "Something failed: #{error?.message}", error + reject error + + ### + Returns a dictionary of all de-serialized records in a given object store. + + @note: To be only used with Cryptobox! + @private + @return [Promise] Promise that will resolve with the de-serialized pre-keys + ### + _load_pre_keys: => + return new Promise (resolve, reject) => + @storage_service.get_all @storage_service.OBJECT_STORE_PREKEYS + .then (pre_keys) => + @logger.log @logger.levels.INFO, "Loaded '#{pre_keys.length}' pre-keys from database" + for pre_key in pre_keys + try + bytes = sodium.from_base64 pre_key.serialised + @prekeys[pre_key.id] = Proteus.keys.PreKey.deserialise bytes.buffer + catch error + @logger.log @logger.levels.ERROR, "Pre-key with primary key '#{pre_key.id}' is corrupt.", error + # TODO: Consider deleting or repairing corrupt data (like broken sessions) + @logger.log @logger.levels.INFO, + "Initialized '#{Object.keys(@prekeys).length}' pre-keys from database", @prekeys + resolve @prekeys + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to load pre-keys from storage: #{error.message}", error + reject error + + _load_sessions: => + return new Promise (resolve, reject) => + @storage_service.get_all @storage_service.OBJECT_STORE_SESSIONS + .then (sessions) => + @sessions_total = sessions.length + amplify.publish z.event.WebApp.APP.UPDATE_INIT, z.string.init_sessions_expectation, true, [@sessions_total] + if @sessions_total > 0 + @logger.log @logger.levels.INFO, "Loaded '#{@sessions_total}' sessions from storage" + @sessions_queue sessions + @sessions_promises = [resolve, reject] + else + @logger.log @logger.levels.INFO, 'No sessions found in storage' + resolve @sessions + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to load sessions from storage: #{error.message}", error + reject error + + ############################################################################### + # Amplify + ############################################################################### + + ### + Get a value for a given primary key from the amplify value store. + + @param primary_key [String] Primary key to retrieve the object for + @return [Promise] Promise that will resolve with the retrieved value + ### + get_value: (primary_key) => + return new Promise (resolve, reject) => + @storage_service.load @storage_service.OBJECT_STORE_AMPLIFY, primary_key + .then (record) -> + if record?.value + resolve record.value + else + reject new Error "Value for primary key '#{primary_key}' not found" + .catch (error) => + @logger.log @logger.levels.ERROR, "Something failed: #{error?.message}", error + reject error + + ### + Save a value in the amplify value store. + + @param primary_key [String] Primary key to save the object with + @param value [value] Object to be stored + @return [Promise] Promise that will resolve with the saved record + ### + save_value: (primary_key, value) => + return @storage_service.save @storage_service.OBJECT_STORE_AMPLIFY, primary_key, value: value + + ### + Closes the database connection. + ### + terminate: -> + @storage_service.terminate() + + ############################################################################### + # Conversation Events + ############################################################################### + + ### + Construct a unique primary key. + + @param conversation_id [String] ID of conversation + @param sender_id [String, undefined] ID of message sender + @param time [String] Time in ISO format to create timestamp from + @return [String] Generated primary key + ### + construct_primary_key: (conversation_id, sender_id = @storage_service.user_id, time) -> + timestamp = new Date(time).getTime() + return "#{conversation_id}@#{sender_id}@#{timestamp}" + + ### + Save an unencrypted conversation event. + + @param event [Object] JSON event to be stored + @return [Promise] Promise that resolves with the stored record + ### + save_unencrypted_conversation_event: (event) -> + return new Promise (resolve, reject) => + primary_key = @construct_primary_key event.conversation, event.from, event.time + + event_object = + raw: event + meta: + timestamp: new Date(event.time).getTime() + version: 1 + + store_name = @storage_service.OBJECT_STORE_CONVERSATION_EVENTS + @storage_service.save store_name, primary_key, event_object + .then (primary_key) -> resolve primary_key + .catch (error) -> reject error + + ### + Save a decrypted conversation event. + + @param primary_key [String] Primary key to save the object with + @param otr_message_event [Object] JSON event to be stored + @param mapped_json [Object] OTR event mapped to its unencrypted counterpart + @return [Promise] Promise that resolves with the stored record + ### + save_decrypted_conversation_event: (primary_key, otr_message_event, mapped_json) -> + return new Promise (resolve, reject) => + event_object = + raw: otr_message_event + meta: + timestamp: new Date(otr_message_event.time).getTime() + version: 1 + mapped: mapped_json + + # We don't need to keep ciphertext once it has been successfully decrypted + event_object.raw.data = undefined + + store_name = @storage_service.OBJECT_STORE_CONVERSATION_EVENTS + @storage_service.save store_name, primary_key, event_object + .then (primary_key) -> resolve primary_key + .catch (error) -> reject error + + ### + Load a conversation event for a given primary key. + + @param primary_key [String] Primary key to save the object with + @return [Promise] Promise that resolves with the retrieved record + ### + load_event_for_conversation: (primary_key) -> + return @storage_service.load @storage_service.OBJECT_STORE_CONVERSATION_EVENTS, primary_key + + ### + Load conversation events by event type. + + @param event_types [Array] Array of event types to match + @return [Promise] Promise that resolves with the retrieved records + ### + load_events_by_types: (event_types) -> + return new Promise (resolve, reject) => + @storage_service.db[@storage_service.OBJECT_STORE_CONVERSATION_EVENTS] + .where 'raw.type' + .anyOf event_types + .sortBy 'raw.time' + .then (records) -> + resolve records + .catch (error) => + @logger.log @logger.levels.ERROR, "Something failed: #{error?.message}", error + reject error + + ############################################################################### + # Identity + ############################################################################### + + ### + Load the identity key pair. + @override + @return [String] Serialized identity key pair + ### + load_identity: -> + return @identity + + ### + Save the identity key pair. + + @override + @param identity [Proteus.keys.IdentityKeyPair] + @return [Promise] Promise that resolves with the saved record + ### + save_identity: (identity) -> + return new Promise (resolve, reject) => + @identity = identity + payload = serialised: sodium.to_base64 new Uint8Array identity.serialise() + + @storage_service.save @storage_service.OBJECT_STORE_KEYS, 'local_identity', payload + .then (primary_key) => + message = "Saved local identity '#{identity.public_key.fingerprint()}' to db '#{@storage_service.db_name}'" + @logger.log @logger.levels.INFO, message, identity + resolve primary_key + .catch (error) -> reject error + + + ############################################################################### + # Pre-keys + ############################################################################### + + ### + Store a pre-key. + + @override + @param prekey [String] Pre-key to be stored + @return [Promise] Promise that resolves with the saved record + ### + add_prekey: (prekey) -> + return new Promise (resolve, reject) => + @prekeys[prekey.key_id] = prekey + payload = + id: prekey.key_id + serialised: sodium.to_base64 new Uint8Array prekey.serialise() + + @storage_service.save @storage_service.OBJECT_STORE_PREKEYS, "#{prekey.key_id}", payload + .then (primary_key) -> resolve primary_key + .catch (error) -> reject error + + ### + Delete a pre-key + + @override + @param prekey_id [String] Primary key to delete pre-key from store + @return [Promise] Promise that resolves with the deleted record + ### + delete_prekey: (prekey_id) -> + return new Promise (resolve, reject) => + delete @prekeys[prekey_id] + + @storage_service.delete @storage_service.OBJECT_STORE_PREKEYS, "#{prekey_id}" + .then (record) -> resolve record + .catch (error) -> reject error + + ### + Load a pre-key. + + @override + @param prekey_id [String] Primary key to retrieve pre-key for + @return [String] Pre-key + ### + load_prekey: (prekey_id) -> + return @prekeys[prekey_id] + + clear_all_stores: => + @storage_service.clear_all_stores() + .then => @logger.log "Cleared database '#{@storage_service.db_name}'" + + ############################################################################### + # Sessions + ############################################################################### + + ### + Delete a session. + + @override + @param session_id [String] ID of session to be deleted from store + @return [Promise] Promise that resolves with the deleted record + ### + delete_session: (session_id) -> + return new Promise (resolve, reject) => + delete @sessions[session_id] + + @storage_service.delete @storage_service.OBJECT_STORE_SESSIONS, session_id + .then (record) -> resolve record + .catch (error) -> reject error + + ### + Load a session. + + @override + @param identity [String] Unused identity + @param session_id [String] ID of session to be retrieved + @return [Object] Session + ### + load_session: (identity, session_id) -> + return @sessions[session_id] + + ### + Save a session. + + @override + @param session_id [String] ID of session to be stored + @param session [Object] Session to be stored + @return [Promise] Promise that resolves with the saved record + ### + save_session: (session_id, session) -> + return new Promise (resolve, reject) => + @sessions[session_id] = session + payload = + id: session_id + serialised: sodium.to_base64 new Uint8Array session.serialise() + + @storage_service.save @storage_service.OBJECT_STORE_SESSIONS, session_id, payload + .then (primary_key) -> + resolve primary_key + .catch (error) -> reject error + + ### + Nuke the database. + ### + delete_everything: => + @logger.log @logger.levels.WARN, "Deleting database '#{@storage_service.db_name}'" + return @storage_service.delete_everything() diff --git a/app/script/storage/StorageService.coffee b/app/script/storage/StorageService.coffee new file mode 100644 index 00000000000..2a870196bcb --- /dev/null +++ b/app/script/storage/StorageService.coffee @@ -0,0 +1,339 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.storage ?= {} + +### +Base class for persistent storage. +### +class z.storage.StorageService + OBJECT_STORE_AMPLIFY: 'amplify' + OBJECT_STORE_CLIENTS: 'clients' + OBJECT_STORE_CONVERSATION_EVENTS: 'conversation_events' + OBJECT_STORE_CONVERSATIONS: 'conversations' + OBJECT_STORE_KEYS: 'keys' + OBJECT_STORE_PREKEYS: 'prekeys' + OBJECT_STORE_SESSIONS: 'sessions' + + constructor: -> + @logger = new z.util.Logger 'z.storage.StorageService', z.config.LOGGER.OPTIONS + + @db = undefined + @db_name = undefined + @user_id = undefined + + return @ + + ############################################################################### + # Initialization + ############################################################################### + + ### + Initialize the IndexedDB for a user. + + @param user_id [String] User ID + @return [Promise] Promise that will resolve with the database name + ### + init: (user_id = @user_id) => + return new Promise (resolve, reject) => + is_permanent = z.storage.get_value z.storage.StorageKey.AUTH.PERSIST + client_type = if is_permanent then z.client.ClientType.PERMANENT else z.client.ClientType.TEMPORARY + + @user_id = user_id + @db_name = "wire@#{z.util.Environment.backend.current}@#{user_id}@#{client_type}" + + # https://github.com/dfahlander/Dexie.js/wiki/Version.stores() + version_1 = + "#{@OBJECT_STORE_AMPLIFY}": '' + "#{@OBJECT_STORE_CLIENTS}": '' + "#{@OBJECT_STORE_CONVERSATION_EVENTS}": ', raw.conversation, raw.time, meta.timestamp' + "#{@OBJECT_STORE_KEYS}": '' + "#{@OBJECT_STORE_PREKEYS}": '' + "#{@OBJECT_STORE_SESSIONS}": '' + + version_2 = + "#{@OBJECT_STORE_AMPLIFY}": '' + "#{@OBJECT_STORE_CLIENTS}": '' + "#{@OBJECT_STORE_CONVERSATION_EVENTS}": ', raw.conversation, raw.time, raw.type, meta.timestamp' + "#{@OBJECT_STORE_KEYS}": '' + "#{@OBJECT_STORE_PREKEYS}": '' + "#{@OBJECT_STORE_SESSIONS}": '' + + version_3 = + "#{@OBJECT_STORE_AMPLIFY}": '' + "#{@OBJECT_STORE_CLIENTS}": '' + "#{@OBJECT_STORE_CONVERSATION_EVENTS}": ', raw.conversation, raw.time, raw.type, meta.timestamp' + "#{@OBJECT_STORE_CONVERSATIONS}": ', id, last_event_timestamp' + "#{@OBJECT_STORE_KEYS}": '' + "#{@OBJECT_STORE_PREKEYS}": '' + "#{@OBJECT_STORE_SESSIONS}": '' + + version_4 = + "#{@OBJECT_STORE_AMPLIFY}": '' + "#{@OBJECT_STORE_CLIENTS}": ', meta.primary_key' + "#{@OBJECT_STORE_CONVERSATION_EVENTS}": ', raw.conversation, raw.time, raw.type, meta.timestamp' + "#{@OBJECT_STORE_CONVERSATIONS}": ', id, last_event_timestamp' + "#{@OBJECT_STORE_KEYS}": '' + "#{@OBJECT_STORE_PREKEYS}": '' + "#{@OBJECT_STORE_SESSIONS}": '' + + @db = new Dexie @db_name + + @db.on 'blocked', => + @logger.log @logger.levels.ERROR, 'Database is blocked' + + # @see https://github.com/dfahlander/Dexie.js/issues/189#issuecomment-196304545 + # @see https://github.com/dfahlander/Dexie.js/wiki/Dexie.UpgradeError + Dexie.Promise.on 'error', (error) => + switch error.name + when Dexie.errnames.Upgrade + @logger.log @logger.levels.ERROR, "Failed to open database '#{@db_name}': #{error.message}", error + else + @logger.log @logger.levels.ERROR, "Database error (#{error.name}): #{error.message}", error + + # @see https://github.com/dfahlander/Dexie.js/wiki/Version.upgrade() + # @see https://github.com/dfahlander/Dexie.js/wiki/WriteableCollection.modify() + @db.version(1).stores version_1 + @db.version(2).stores version_2 + @db.version(3).stores version_3 + @db.version(4).stores version_4 + .upgrade (transaction) => + @logger.log @logger.levels.WARN, 'Database upgrade to version 4 needed', transaction + transaction[@OBJECT_STORE_CLIENTS].toCollection().modify (client) => + client.meta = + is_verified: true + primary_key: 'local_identity' + @logger.log 'Updated client', client + @db.version(5).stores version_4 + @db.version(6).stores version_4 + .upgrade (transaction) => + @logger.log @logger.levels.WARN, 'Database upgrade to version 6 needed', transaction + transaction[@OBJECT_STORE_CONVERSATIONS].toCollection().eachKey (key) => + @db[@OBJECT_STORE_CONVERSATIONS].update key, {id: key} + transaction[@OBJECT_STORE_SESSIONS].toCollection().eachKey (key) => + @db[@OBJECT_STORE_SESSIONS].update key, {id: key} + transaction[@OBJECT_STORE_PREKEYS].toCollection().eachKey (key) => + @db[@OBJECT_STORE_PREKEYS].update key, {id: key} + + @db.open() + .then => + @logger.log @logger.levels.LEVEL_1, "Storage Service initialized with database '#{@db_name}'" + resolve @db_name + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to initialize database '#{@db_name}' for Storage Service" + reject error + + + ############################################################################### + # Interactions + ############################################################################### + + ### + Removes persisted data. + + @param store_name [String] Name of the object store + @param primary_key [String] Primary key + @return [Promise] Promise that will resolve when the object is deleted + ### + delete: (store_name, primary_key) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name].delete primary_key + .then => + @logger.log "Deleted '#{primary_key}' from object store '#{store_name}'" + resolve primary_key + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to delete '#{primary_key}' from store '#{store_name}'", error + reject error + else + reject new Error "Data store '#{store_name}' not found" + + clear_all_stores: => + promises = (@delete_store store_name for store_name of @db._dbSchema) + return Promise.all promises + + ### + @return [Promise] + ### + delete_store: (store_name) => + @logger.log "Clearing object store '#{store_name}' in database '#{@db_name}'" + return @db[store_name].clear() + + ### + Delete the IndexedDB with all its stores. + @return [Promise] Promise that will resolve if a database is found and cleared + ### + delete_everything: => + return new Dexie.Promise (resolve, reject) => + if @db? + @db.delete() + .then => + @logger.log @logger.levels.INFO, "Clearing IndexedDB '#{@db_name}' successful" + resolve true + .catch (error) => + @logger.log @logger.levels.ERROR, "Clearing IndexedDB '#{@db_name}' failed" + reject error + else + @logger.log @logger.levels.ERROR, "IndexedDB '#{@db_name}' not found" + resolve true + + ### + Returns an array of all records for a given object store. + + @param store_name [String] Name of object store + @return [Promise] Promise that will resolve with the records from the object store + ### + get_all: (store_name) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name] + .toArray() + .then (records) -> resolve records + .catch (error) => + @logger.log @logger.levels.ERROR, "Could not load objects from store '#{store_name}'", error + reject error + else + reject new Error "Data store '#{store_name}' not found" + + ### + Returns an array of all keys in a given object store. + + @param store_name [String] Name of object store + @return [Promise] Promise that will resolve with the keys of the object store + ### + get_keys: (store_name, regex_string) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name] + .toCollection() + .keys() + .then (keys) -> + if regex_string is undefined + resolve keys + else + regex = new RegExp regex_string, 'igm' + accepted_keys = keys.filter (key) -> key.match regex + resolve accepted_keys + .catch (error) -> reject error + else + reject new Error "Data store '#{store_name}' not found" + + ### + Loads persisted data via a promise. + + @note If a key cannot be found, it resolves and returns "undefined". + + @param store_name [String] Name of object store + @param primary_key [String] Primary key of object to be retrieved + @return [Promise] Promise that will resolve with the record matching the primary key + ### + load: (store_name, primary_key) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name].get primary_key + .then (record) -> resolve record + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to load '#{primary_key}' from store '#{store_name}'", error + reject error + else + reject new Error "Data store '#{store_name}' not found" + + ### + Loads all objects from an object store and returns them with their keys and values. + + @example Return value: + { + "key_1": { + "property_1": value_1, + "property_2": value_2 + }, + "key_2": { + "property_1": value_1, + "property_2": value_2 + }, + ... + } + + @param store_name [String] Name of object store + @return [Promise] Promise which resolves with an object containing all keys and values + ### + load_all: (store_name, regex_string) => + return new Dexie.Promise (resolve, reject) => + data = {} + + @get_keys store_name, regex_string + .then (keys) => + promises = keys.map (key) => + return new Promise (resolve, reject) => + @load store_name, key + .then (value) -> + data[key] = value + resolve undefined + .catch (error) -> reject error + return Promise.all promises + .then -> resolve data + .catch (error) -> reject error + + ### + Saves objects in the local database. + + @param store_name [String] Name of object store where to save the object + @param primary_key [String] Primary key which should be used to store the object + @return [Promise] Promise that will resolve with the primary key of the persisted object + ### + save: (store_name, primary_key, entity) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name].put entity, primary_key + .then (key) -> resolve key + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to put '#{primary_key}' into store '#{store_name}'", error + reject error + else + reject new Error "Data store '#{store_name}' not found" + + ### + Closes the database. This operation completes immediately and there is no returned Promise. + @see https://github.com/dfahlander/Dexie.js/wiki/Dexie.close() + ### + terminate: -> + @logger.log "Closing database connection with '#{@db.name}'" + @db.close() + + ### + Update previously persisted data via a promise. + + @param store_name [String] Name of object store + @param primary_key [String] Primary key of object to be updated + @param changes [Object] Object containing the key paths to each property you want to change + @return [Promise] Promise with the number of updated records (1 if an object was updated, otherwise 0). + ### + update: (store_name, primary_key, changes) => + return new Dexie.Promise (resolve, reject) => + if @db[store_name]? + @db[store_name].update primary_key, changes + .then (number_of_updates) => + @logger.log @logger.levels.INFO, + "Updated #{number_of_updates} record(s) with key '#{primary_key}' in store '#{store_name}'", changes + resolve number_of_updates + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to update '#{primary_key}' in store '#{store_name}'", error + reject error + else + reject new Error "Data store '#{store_name}' not found" diff --git a/app/script/system_notification/SystemNotificationRepository.coffee b/app/script/system_notification/SystemNotificationRepository.coffee new file mode 100644 index 00000000000..77a3a4b656c --- /dev/null +++ b/app/script/system_notification/SystemNotificationRepository.coffee @@ -0,0 +1,476 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.SystemNotification ?= {} + +### +System notification repository to trigger browser and audio notifications. + +@see https://developer.mozilla.org/en/docs/Web/API/notification +@see http://www.w3.org/TR/notifications +### +class z.SystemNotification.SystemNotificationRepository + @::EVENTS_TO_NOTIFY = [ + z.message.SuperType.CALL + z.message.SuperType.CONTENT + z.message.SuperType.MEMBER + z.message.SuperType.PING + z.message.SuperType.SYSTEM + ] + + ### + Construct a new System Notification Repository. + @param conversation_repository [z.conversation.ConversationService] Repository for all conversation interactions + ### + constructor: (@conversation_repository) -> + @logger = new z.util.Logger 'z.SystemNotification.SystemNotificationRepository', z.config.LOGGER.OPTIONS + + @ask_for_permission = true + @notifications = [] + + @muted = true + @subscribe_to_events() + + subscribe_to_events: => + amplify.subscribe z.event.WebApp.EVENT.NOTIFICATION_HANDLING_STATE, @set_muted_state + amplify.subscribe z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, @, @notify + amplify.subscribe z.event.WebApp.SYSTEM_NOTIFICATION.REMOVE_READ, @, @remove_read_notifications + amplify.subscribe z.event.WebApp.SYSTEM_NOTIFICATION.REQUEST_PERMISSION, @, @should_request + + ### + Display browser notification and play sound notification. + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + ### + notify: (conversation_et, message_et) => + return if @muted or message_et.super_type not in @EVENTS_TO_NOTIFY + @_notify_sound conversation_et, message_et + @_notify_banner conversation_et, message_et + + ### + Remove notifications from the queue that are no longer unread + ### + remove_read_notifications: => + for notification in @notifications + return if not notification.data? + conversation_id = notification.data.conversation_id + message_id = notification.data.message_id + if @conversation_repository.is_message_read conversation_id, message_id + notification.close() + @logger.log @logger.levels.INFO, "Removed read notification for '#{message_id}' in '#{conversation_id}'." + + ### + Request browser permission for notifications. + @param on_permission_granted [Function] Function to be called when permission is granted + @param on_permission_denied [Function] Function to be called when permission is denied + ### + request_permission: (on_permission_granted, on_permission_denied) -> + return if not z.util.Environment.browser.supports.notifications + if window.Notification.permission is z.util.BrowserPermissionType.DEFAULT + amplify.publish z.event.WebApp.WARNINGS.SHOW, z.ViewModel.WarningType.REQUEST_NOTIFICATION + # Note: The callback will be only triggered in Chrome. + # If you ignore a permission request on Firefox, then the callback will not be triggered. + window.Notification.requestPermission? (permission) -> + amplify.publish z.event.WebApp.WARNINGS.DISMISS, z.ViewModel.WarningType.REQUEST_NOTIFICATION + if permission is z.util.BrowserPermissionType.GRANTED + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.EventName.PERMISSION.ALLOW_NOTIFICATIONS, value: 'allow' + on_permission_granted?() + else + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.REQUEST_PERMISSION, false + amplify.publish z.event.WebApp.ANALYTICS.EVENT, + z.tracking.EventName.PERMISSION.ALLOW_NOTIFICATIONS, value: 'block' + on_permission_denied?() + + ### + Set the muted state. + @note Temporarily mute notifications on recovery from Notification Stream + @param muted [Boolean] Are sounds and notifications muted + ### + set_muted_state: (handling_notifications) => + @muted = handling_notifications + @logger.log @logger.levels.INFO, "Setting muted state to: #{handling_notifications}" + + ### + @param should_request [Boolean] True, when permission should be requested + ### + should_request: (should_request) -> + @ask_for_permission = should_request + + ### + Sending the browser notification. + + @param notification_content [Object] + @option notification_content [String] title + @option notification_content [Object] options + @option notification_content [Function] trigger + @option notification_content [Integer] timeout + ### + show: (notification_content) -> + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.SHOW, notification_content + @_show_notification notification_content + + ### + Check for browser permission if we have not yet asked. + + @param success_callback [Function] Function to be called if permission is granted + ### + _check_permission: (success_callback) => + if @ask_for_permission is true + switch window.Notification.permission + when z.util.BrowserPermissionType.DEFAULT + @request_permission success_callback + when z.util.BrowserPermissionType.GRANTED + success_callback() + + ### + Creates the notification body for calls. + @private + @return [String] Notification message body + ### + _create_body_call: (message_et) -> + if message_et.is_call_activation() + return z.localization.Localizer.get_text z.string.system_notification_voice_channel_activate + else if message_et.is_call_deactivation() + return if message_et.finished_reason isnt z.calling.enum.CallFinishedReason.MISSED + return z.localization.Localizer.get_text z.string.system_notification_voice_channel_deactivate + + ### + Creates the notification body for text messages and pictures. + + @private + @param message_et [z.entity.NormalMessage] Normal message entity + @return [String] Notification message body + #### + _create_body_content: (message_et) -> + if message_et.has_asset_text() + for asset_et in message_et.assets() when asset_et.is_text() + return z.util.trunc_text asset_et.text, z.config.BROWSER_NOTIFICATION.BODY_LENGTH + else if message_et.has_asset_medium_image() + return z.localization.Localizer.get_text z.string.system_notification_asset_add + else if message_et.has_asset() + asset_et = message_et.assets()[0] + return z.localization.Localizer.get_text z.string.system_notification_shared_audio if asset_et.is_audio() + return z.localization.Localizer.get_text z.string.system_notification_shared_video if asset_et.is_video() + return z.localization.Localizer.get_text z.string.system_notification_shared_file if asset_et.is_file() + + ### + Creates the notification body for a renamed conversation. + + @private + @param message_et [z.entity.RenameMessage] Rename message entity + @return [String] Notification message body + ### + _create_body_conversation_rename: (message_et) -> + return z.localization.Localizer.get_text { + id: z.string.system_notification_conversation_rename + replace: [ + {placeholder: '%s.first_name', content: message_et.user().first_name()} + {placeholder: '%name', content: message_et.name} + ] + } + + ### + Creates the notification body for people being added to a group conversation. + + @private + @param message_et [z.entity.Message] Member message entity + @return [String] Notification message body + ### + _create_body_member_join: (message_et) -> + if message_et.user_ets().length is 1 + return z.localization.Localizer.get_text { + id: z.string.system_notification_member_join_one + replace: [ + {placeholder: '%s.first_name', content: message_et.user().first_name()} + { + placeholder: '%@.first_name' + content: z.util.get_first_name message_et.user_ets()[0], z.string.Declension.ACCUSATIVE + } + ] + } + return z.localization.Localizer.get_text { + id: z.string.system_notification_member_join_many + replace: [ + {placeholder: '%s.first_name', content: message_et.user().first_name()} + {placeholder: '%no', content: message_et.user_ids().length} + ] + } + + + ### + Creates the notification body for people being removed from or leaving a group conversation. + + @private + @param message_et [z.entity.MemberMessage] Member message entity + @return [String] Notification message body + ### + _create_body_member_leave: (message_et) -> + if message_et.user_ets().length is 1 + if message_et.user_ets()[0] is message_et.user() + return z.localization.Localizer.get_text { + id: z.string.system_notification_member_leave_left + replace: {placeholder: '%s.first_name', content: message_et.user().first_name()} + } + return z.localization.Localizer.get_text { + id: z.string.system_notification_member_leave_removed_one + replace: [ + {placeholder: '%s.first_name', content: message_et.user().first_name()} + { + placeholder: '%@.first_name' + content: z.util.get_first_name message_et.user_ets()[0], z.string.Declension.ACCUSATIVE + } + ] + } + return z.localization.Localizer.get_text { + id: z.string.system_notification_member_leave_removed_many + replace: [ + {placeholder: '%s.first_name', content: message_et.user().first_name()} + {placeholder: '%no', content: message_et.user_ets().length} + ] + } + + ### + Selects the type of system message that the notification body needs to be created for. + + @private + @param message_et [z.entity.MemberMessage] Member message entity + @param is_group_conversation [Boolean] Is a group conversation + @return [String] Notification message body + ### + _create_body_member_update: (message_et, is_group_conversation) -> + switch message_et.member_message_type + when z.message.SystemMessageType.NORMAL + return if not is_group_conversation + if message_et.type is z.event.Backend.CONVERSATION.MEMBER_JOIN + return @_create_body_member_join message_et + else if message_et.type is z.event.Backend.CONVERSATION.MEMBER_LEAVE + return @_create_body_member_leave message_et + when z.message.SystemMessageType.CONNECTION_ACCEPTED + return z.localization.Localizer.get_text z.string.system_notification_connection_accepted + when z.message.SystemMessageType.CONNECTION_REQUEST + return z.localization.Localizer.get_text z.string.system_notification_connection_request + when z.message.SystemMessageType.CONVERSATION_CREATE + return z.localization.Localizer.get_text { + id: z.string.system_notification_conversation_create + replace: {placeholder: '%s.first_name', content: message_et.user().first_name()} + } + + ### + Creates the notification body for ping and hot-ping. + + @private + @param message_et [z.entity.PingMessage] Ping message entity + @return [String] Notification message body + ### + _create_body_ping: (message_et) -> + return z.localization.Localizer.get_text z.string.system_notification_ping + + ### + Selects the type of system message that the notification body needs to be created for. + + @private + @param message_et [z.entity.MemberMessage] Member message entity + @param conversation_type [z.conversation.ConversationType] Type of the conversation + @return [String] Notification message body + ### + _create_body_system: (message_et) => + switch message_et.system_message_type + when z.message.SystemMessageType.CONVERSATION_RENAME + return @_create_body_conversation_rename + ### + Selects the type of message that the notification body needs to be created for. + + @private + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + @return [String] Notification message body + ### + _create_options_body: (conversation_et, message_et) => + switch message_et.super_type + when z.message.SuperType.CALL + return @_create_body_call message_et + when z.message.SuperType.CONTENT + return @_create_body_content message_et + when z.message.SuperType.MEMBER + return @_create_body_member_update message_et, conversation_et.is_group?() + when z.message.SuperType.PING + return @_create_body_ping message_et + when z.message.SuperType.SYSTEM + return @_create_body_conversation_rename message_et + + ### + Creates the notification data to help check its content. + + @private + @param input [z.entity.Conversation, z.entity.Connection] Information to grab the conversation ID from + @param message_et [z.entity.Message] Message entity + @return [String] Notification message data + ### + _create_options_data: (input, message_et) -> + return {} = + conversation_id: input.id or input.conversation_id + message_id: message_et.id + + ### + Creates the notification tag. + + @private + @param input [z.entity.Conversation, z.entity.Connection] Information to create the tag from + @return [String] Notification message tag + ### + _create_options_tag: (input) -> + return input.id or input.conversation_id + + ### + Creates the notification title. + + @private + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + @return [String] Notification message title + ### + _create_title: (conversation_et, message_et) -> + if conversation_et.display_name?() + if conversation_et.is_group() + return z.util.trunc_text "#{message_et.user().first_name()} in #{conversation_et.display_name()}", z.config.BROWSER_NOTIFICATION.TITLE_LENGTH, false + return z.util.trunc_text conversation_et.display_name(), z.config.BROWSER_NOTIFICATION.TITLE_LENGTH, false + if not message_et.user() + Raygun.send new Error 'Message does not contain user info' + else + return z.util.trunc_text message_et.user().name(), z.config.BROWSER_NOTIFICATION.TITLE_LENGTH, false + + ### + Creates the notification trigger. + + @private + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + @return [Function] Function to be called when notification is clicked + ### + _create_trigger: (conversation_et, message_et) -> + if message_et.is_member() + switch message_et.member_message_type + when z.message.SystemMessageType.CONNECTION_ACCEPTED + return -> amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et.conversation_id + when z.message.SystemMessageType.CONNECTION_REQUEST + return -> amplify.publish z.event.WebApp.PENDING.SHOW + return -> amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + + ### + Creates the browser notification and sends it. + + @private + @see https://developer.mozilla.org/en/docs/Web/API/notification#Parameters + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + ### + _notify_banner: (conversation_et, message_et) -> + return if z.util.Environment.browser.supports.notifications is false + return if window.Notification.permission is z.util.BrowserPermissionType.DENIED + return if document.hasFocus() + return if message_et.user()?.is_me + return if conversation_et.is_muted?() + + notification_content = + title: @_create_title conversation_et, message_et + options: + body: @_create_options_body conversation_et, message_et + data: @_create_options_data conversation_et, message_et + icon: if z.util.Environment.electron and z.util.Environment.os.mac then '' else window.notification_icon or '/image/logo/notification.png' + tag: @_create_options_tag conversation_et + silent: true #@note When Firefox supports this we can remove the fix for WEBAPP-731 + timeout: z.config.BROWSER_NOTIFICATION.TIMEOUT + trigger: @_create_trigger conversation_et, message_et + + return if not notification_content.options.body? + + @_check_permission => @show notification_content + + ### + Plays the sound from the audio repository. + + @private + @param conversation_et [z.entity.Conversation] Conversation entity + @param message_et [z.entity.Message] Message entity + ### + _notify_sound: (conversation_et, message_et) -> + return if conversation_et.is_muted?() + return if not document.hasFocus() and z.util.Environment.browser.firefox and z.util.Environment.os.mac + switch message_et.super_type + when z.message.SuperType.CONTENT + return if message_et.user().is_me + return if message_et.has_asset_preview_image() + unless document.hasFocus() and conversation_et.id is @conversation_repository.active_conversation()?.id + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.NEW_MESSAGE + when z.message.SuperType.PING + if message_et.user().is_me + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.OUTGOING_PING + else + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.INCOMING_PING + + ### + Sending the browser notification. + + @private + @param notification_content [Object] + @option notification_content [String] title + @option notification_content [Object] options + @option notification_content [Function] trigger + @option notification_content [Integer] timeout + ### + _show_notification: (notification_content) -> + ### + @note Notification.data is only supported on Chrome + @see https://developer.mozilla.org/en-US/docs/Web/API/Notification/data + ### + @remove_read_notifications() + notification = new window.Notification notification_content.title, notification_content.options + conversation_id = notification_content.options.data.conversation_id + message_id = notification_content.options.data.message_id + timeout_trigger_id = undefined + notification.onclick = => + notification_content.trigger() + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.CLICK + @logger.log @logger.levels.INFO, "Notification for '#{message_id} in '#{conversation_id}' closed by click." + window.focus() + notification.close() + notification.onclose = => + clearTimeout timeout_trigger_id + @notifications.splice @notifications.indexOf(notification), 1 + @logger.log @logger.levels.INFO, "Removed notification for '#{message_id}' in '#{conversation_id}' locally." + notification.onerror = => + @logger.log @logger.levels.ERROR, "Notification for '#{message_id}' in '#{conversation_id}' closed by error." + notification.close() + notification.onshow = => + timeout_trigger_id = setTimeout => + @logger.log @logger.levels.INFO, "Notification for '#{message_id}' in '#{conversation_id}' closed by timeout." + notification.close() + , notification_content.timeout + + @notifications.push notification + @logger.log @logger.levels.INFO, "Added notification for '#{message_id}' in '#{conversation_id}' to queue." + window.onunload = => + for notification in @notifications + notification.close() + + return if not notification.data? + conversation_id = notification.data.conversation_id + message_id = notification.data.message_id + @logger.log @logger.levels.INFO, "Notification for '#{message_id}' in '#{conversation_id}' closed by redirect." diff --git a/app/script/telemetry/app_init/AppInitStatistics.coffee b/app/script/telemetry/app_init/AppInitStatistics.coffee new file mode 100644 index 00000000000..39c7346be99 --- /dev/null +++ b/app/script/telemetry/app_init/AppInitStatistics.coffee @@ -0,0 +1,49 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.app_init ?= {} + +class z.telemetry.app_init.AppInitStatistics + constructor: -> + @logger = new z.util.Logger 'z.telemetry.app_init.AppInitStatistics', z.config.LOGGER.OPTIONS + + amplify.subscribe z.event.WebApp.TELEMETRY.BACKEND_REQUESTS, @update_backend_requests + + add: (statistic, value, bucket_size) => + if bucket_size and _.isNumber value + buckets = Math.floor(value / bucket_size) + if value % bucket_size then 1 else 0 + @[statistic] = if value is 0 then 0 else bucket_size * buckets + else + @[statistic] = value + + get: => + statistics = {} + statistics[key] = value for key, value of @ when _.isNumber(value) or _.isString value + return statistics + + log: => + @logger.log @logger.levels.DEBUG, 'App initialization statistics' + for key, value of @ when _.isNumber(value) or _.isString value + placeholder_key = Array(Math.max 17 - key.length, 1).join ' ' + placeholder_value = Array(Math.max 11 - value.toString().length, 1).join ' ' + @logger.log @logger.levels.INFO, "#{placeholder_key}'#{key}':#{placeholder_value}#{value}" + + update_backend_requests: (number_of_requests) => + @[z.telemetry.app_init.AppInitStatisticsValue.BACKEND_REQUESTS] = number_of_requests diff --git a/app/script/telemetry/app_init/AppInitStatisticsValue.coffee b/app/script/telemetry/app_init/AppInitStatisticsValue.coffee new file mode 100644 index 00000000000..2d4d469d527 --- /dev/null +++ b/app/script/telemetry/app_init/AppInitStatisticsValue.coffee @@ -0,0 +1,30 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.app_init ?= {} + +z.telemetry.app_init.AppInitStatisticsValue = + BACKEND_REQUESTS: 'backend_requests' + CLIENT_TYPE: 'client_type' + CLIENTS: 'clients' + CONNECTIONS: 'connections' + CONVERSATIONS: 'conversations' + NOTIFICATIONS: 'notifications' + SESSIONS: 'sessions' diff --git a/app/script/telemetry/app_init/AppInitTelemetry.coffee b/app/script/telemetry/app_init/AppInitTelemetry.coffee new file mode 100644 index 00000000000..81db703a85a --- /dev/null +++ b/app/script/telemetry/app_init/AppInitTelemetry.coffee @@ -0,0 +1,56 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.app_init ?= {} + +class z.telemetry.app_init.AppInitTelemetry + constructor: -> + @logger = new z.util.Logger 'z.telemetry.app_init.AppInitTelemetry', z.config.LOGGER.OPTIONS + @timings = new z.telemetry.app_init.AppInitTimings() + @statistics = new z.telemetry.app_init.AppInitStatistics() + + add_statistic: (statistic, value, bucket_size) => + @statistics.add statistic, value, bucket_size + + get_statistics: => + @statistics.get() + + get_timings: => + @timings.get() + + log_statistics: => + @statistics.log() + + log_timings: => + @timings.log() + + report: => + statistics = @get_statistics() + statistics.loading_time = @timings.get_app_load() + statistics.app_version = z.util.Environment.version false + @logger.log @logger.levels.DEBUG, 'App initialization telemetry' + @logger.log @logger.levels.INFO, "App version '#{statistics.app_version}' initialized within #{statistics.loading_time}s" + @log_statistics() + @log_timings() + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.TELEMETRY.APP_INITIALIZATION, statistics + + time_step: (step) => + @timings.time_step step diff --git a/app/script/telemetry/app_init/AppInitTimings.coffee b/app/script/telemetry/app_init/AppInitTimings.coffee new file mode 100644 index 00000000000..f59143985ee --- /dev/null +++ b/app/script/telemetry/app_init/AppInitTimings.coffee @@ -0,0 +1,46 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.app_init ?= {} + +class z.telemetry.app_init.AppInitTimings + constructor: -> + @logger = new z.util.Logger 'z.telemetry.AppInitTimings', z.config.LOGGER.OPTIONS + @init = window.performance.now() + + get: => + timings = {} + timings[key] = value for key, value of @ when key.toString() isnt 'init' and _.isNumber value + return timings + + get_app_load: => + bucket_size = 10 + return bucket_size * (Math.floor(@[z.telemetry.app_init.AppInitTimingsStep.SHOWING_UI] / bucket_size / 1000) + 1) + + log: => + @logger.log @logger.levels.DEBUG, 'App initialization step durations' + for key, value of @ when key.toString() isnt 'init' and _.isNumber value + placeholder_key = Array(Math.max 27 - key.length, 1).join ' ' + placeholder_value = Array(Math.max 6 - value.toString().length, 1).join ' ' + @logger.log @logger.levels.INFO, "#{placeholder_key}'#{key}':#{placeholder_value}#{value}ms" + + time_step: (step) => + if not @[step] + @[step] = window.parseInt window.performance.now() - @init diff --git a/app/script/telemetry/app_init/AppInitTimingsStep.coffee b/app/script/telemetry/app_init/AppInitTimingsStep.coffee new file mode 100644 index 00000000000..0098bd64ba9 --- /dev/null +++ b/app/script/telemetry/app_init/AppInitTimingsStep.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.app_init ?= {} + +z.telemetry.app_init.AppInitTimingsStep = + RECEIVED_ACCESS_TOKEN: 'received_access_token' + INITIALIZED_PROTO_MESSAGES: 'initialized_proto_messages' + RECEIVED_SELF_USER: 'received_self_user' + INITIALIZED_STORAGE: 'initialized_storage' + INITIALIZED_CRYPTOGRAPHY: 'initialized_cryptography' + VALIDATED_CLIENT: 'validated_client' + RECEIVED_USER_DATA: 'received_user_data' + UPDATED_FROM_NOTIFICATIONS: 'updated_from_notifications' + APP_PRE_LOADED: 'app_pre_loaded' + SHOWING_UI: 'showing_ui' + APP_LOADED: 'app_loaded' + UPDATED_CONVERSATIONS: 'updated_conversations' diff --git a/app/script/telemetry/calling/AudioStreamStats.coffee b/app/script/telemetry/calling/AudioStreamStats.coffee new file mode 100644 index 00000000000..f056bbaa09f --- /dev/null +++ b/app/script/telemetry/calling/AudioStreamStats.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +class z.telemetry.calling.AudioStreamStats extends z.telemetry.calling.MediaStreamStats + constructor: (timestamp) -> + super timestamp + @media_type = z.calling.enum.MediaType.AUDIO + @volume_received = 0 + @volume_sent = 0 diff --git a/app/script/telemetry/calling/CallSetupSteps.coffee b/app/script/telemetry/calling/CallSetupSteps.coffee new file mode 100644 index 00000000000..a93f4370139 --- /dev/null +++ b/app/script/telemetry/calling/CallSetupSteps.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +z.telemetry.calling.CallSetupSteps = + ICE_CONNECTION_COMPLETED: 'ice_connection_completed' + ICE_CONNECTION_CONNECTED: 'ice_connection_connected' + ICE_CONNECTION_CHECKING: 'ice_connection_checking' + ICE_GATHERING_COMPLETED: 'ice_gathering_completed' + ICE_GATHERING_STARTED: 'ice_gathering_started' + FLOW_RECEIVED: 'flow_received' + LOCAL_SDP_CREATED: 'local_sdp_created' + LOCAL_SDP_SEND: 'local_sdp_send' + LOCAL_SDP_SET: 'local_sdp_set' + PEER_CONNECTION_CREATED: 'peer_connection_created' + REMOTE_SDP_RECEIVED: 'remote_sdp_received' + REMOTE_SDP_SET: 'remote_sdp_set' + STARTED: 'started' + STATE_PUT: 'state_put' + STREAM_RECEIVED: 'stream_received' + STREAM_REQUESTED: 'stream_requested' diff --git a/app/script/telemetry/calling/CallSetupStepsOrder.coffee b/app/script/telemetry/calling/CallSetupStepsOrder.coffee new file mode 100644 index 00000000000..33e827ba57e --- /dev/null +++ b/app/script/telemetry/calling/CallSetupStepsOrder.coffee @@ -0,0 +1,57 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +z.telemetry.calling.CallSetupStepsOrder = + ANSWER: [ + z.telemetry.calling.CallSetupSteps.STREAM_REQUESTED + z.telemetry.calling.CallSetupSteps.STREAM_RECEIVED + z.telemetry.calling.CallSetupSteps.STATE_PUT + z.telemetry.calling.CallSetupSteps.FLOW_RECEIVED + z.telemetry.calling.CallSetupSteps.PEER_CONNECTION_CREATED + z.telemetry.calling.CallSetupSteps.REMOTE_SDP_RECEIVED + z.telemetry.calling.CallSetupSteps.REMOTE_SDP_SET + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_CREATED + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SET + z.telemetry.calling.CallSetupSteps.ICE_GATHERING_STARTED + z.telemetry.calling.CallSetupSteps.ICE_GATHERING_COMPLETED + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SEND + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CHECKING + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CONNECTED + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_COMPLETED + ] + OFFER: [ + z.telemetry.calling.CallSetupSteps.STREAM_REQUESTED + z.telemetry.calling.CallSetupSteps.STREAM_RECEIVED + z.telemetry.calling.CallSetupSteps.STATE_PUT + z.telemetry.calling.CallSetupSteps.FLOW_RECEIVED + z.telemetry.calling.CallSetupSteps.PEER_CONNECTION_CREATED + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_CREATED + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SET + z.telemetry.calling.CallSetupSteps.ICE_GATHERING_STARTED + z.telemetry.calling.CallSetupSteps.ICE_GATHERING_COMPLETED + z.telemetry.calling.CallSetupSteps.LOCAL_SDP_SEND + z.telemetry.calling.CallSetupSteps.REMOTE_SDP_RECEIVED + z.telemetry.calling.CallSetupSteps.REMOTE_SDP_SET + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CHECKING + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CONNECTED + z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_COMPLETED + ] diff --git a/app/script/telemetry/calling/CallSetupTimings.coffee b/app/script/telemetry/calling/CallSetupTimings.coffee new file mode 100644 index 00000000000..b0765f4dbde --- /dev/null +++ b/app/script/telemetry/calling/CallSetupTimings.coffee @@ -0,0 +1,68 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +class z.telemetry.calling.CallSetupTimings + constructor: (@call_id) -> + @logger = new z.util.Logger 'z.telemetry.calling.CallSetupTimings', z.config.LOGGER.OPTIONS + @is_answer = false + @flow_id = undefined + + @started = window.performance.now() + @stream_requested = 0 + @stream_received = 0 + @state_put = 0 + @flow_received = 0 + @peer_connection_created = 0 + @remote_sdp_received = 0 + @remote_sdp_set = 0 + @local_sdp_created = 0 + @local_sdp_send = 0 + @local_sdp_set = 0 + @ice_gathering_started = 0 + @ice_gathering_completed = 0 + @ice_connection_checking = 0 + @ice_connection_connected = 0 + @ice_connection_completed = 0 + + get: => + timings = {} + for step in @_steps_order() + timings[step] = @[step] + return timings + + time_step: (step) -> + if @[step] is 0 + @[step] = window.parseInt window.performance.now() - @started + + log: => + @logger.log @logger.levels.INFO, "Call setup duration for flow ID '#{@flow_id}' of call ID '#{@call_id}'" + for step in @_steps_order() + placeholder_key = Array(Math.max 26 - step.length, 1).join ' ' + placeholder_value = Array(Math.max 6 - @[step].toString().length, 1).join ' ' + @logger.log @logger.levels.INFO, "Step#{placeholder_key}'#{step}':#{placeholder_value}#{@[step]}ms" + + _steps_order: -> + if @is_answer + order = z.telemetry.calling.CallSetupStepsOrder.ANSWER + else + order = z.telemetry.calling.CallSetupStepsOrder.OFFER + return order diff --git a/app/script/telemetry/calling/CallTelemetry.coffee b/app/script/telemetry/calling/CallTelemetry.coffee new file mode 100644 index 00000000000..c661dd1ebb3 --- /dev/null +++ b/app/script/telemetry/calling/CallTelemetry.coffee @@ -0,0 +1,198 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +# Call traces entity. +class z.telemetry.calling.CallTelemetry + constructor: -> + @logger = new z.util.Logger 'z.telemetry.calling.CallTelemetry', z.config.LOGGER.OPTIONS + + @sessions = {} + @requests = {} + @traces = {} + + + ############################################################################### + # Call traces + ############################################################################### + + ### + Add an event to the call trace. + @param event [JSON] Backend event + ### + trace_event: (event) => + @traces[event.conversation] ?= [] + + timing_incoming = Date.now() + + @traces[event.conversation].push { + from: 'backend' + to: 'us' + transport: 'WebSocket' + response: + payload: event + timestamp: timing_incoming + timestamp_iso_8601: new Date(timing_incoming).toISOString() + } + + ### + Add a backend response to the debug trace. + @param conversation_id [String] Conversation ID of call + @param jqXHR [jQuery XMLHttpRequest] jQuery object of backend response + ### + trace_request: (conversation_id, jqXHR) -> + timestamp_incoming = jqXHR.wire.responded.getTime() + timestamp_outgoing = jqXHR.wire.requested.getTime() + + request_duration = timestamp_incoming - timestamp_outgoing + request_type = "#{jqXHR.wire.original_request_options.type} #{jqXHR.wire.original_request_options.api_endpoint}" + @requests[request_type] ?= [] + @requests[request_type].push request_duration + hits = @requests[request_type].length + average = z.util.Statistics.average @requests[request_type] + standard_deviation = z.util.Statistics.standard_deviation @requests[request_type], average + @logger.log @logger.levels.INFO, "Request #{request_type} took #{request_duration}ms" + @logger.log @logger.levels.INFO, "# of requests #{hits} - Avg: #{average}ms | SD: #{standard_deviation}" + + trace_payload = {} + if jqXHR.wire.original_request_options.data + trace_payload = JSON.parse z.util.types.convert_array_buffer_to_string jqXHR.wire.original_request_options.data + + @traces[conversation_id] ?= [] + @traces[conversation_id].push { + from: 'us' + to: 'backend' + transport: 'REST' + request: + method: jqXHR.wire.original_request_options.type + url: jqXHR.wire.original_request_options.url + payload: trace_payload + timestamp: timestamp_outgoing + timestamp_iso_8601: new Date(timestamp_outgoing).toISOString() + response: + status: + code: jqXHR.status + text: jqXHR.statusText + payload: jqXHR.responseJSON + timestamp: timestamp_incoming + timestamp_iso_8601: new Date(timestamp_incoming).toISOString() + } + + + ############################################################################### + # Sessions + ############################################################################### + + # Force log last call session IDs. + log_sessions: => + @logger.force_log 'Your last session IDs:' + sorted_sessions = z.util.sort_object_by_keys @sessions, true + @logger.force_log tracking_info.to_string() for session_id, tracking_info of sorted_sessions + return sorted_sessions + + ### + Track session ID. + @param conversation_id [String] ID of conversation for call session + @param event [JSON] Call event from backend + ### + track_session: (conversation_id, event) => + @sessions[event.session] = new z.calling.CallTrackingInfo { + conversation_id: conversation_id + session_id: event.session + } + + + ############################################################################### + # Error reporting + ############################################################################### + + ### + Report an error to Raygun. + @param description [String] Error description + @param custom_date [Object] Custom data passed into the report + ### + report_error: (description, passed_error) -> + raygun_error = new Error description + + if passed_error + custom_data = + error: passed_error + raygun_error.stack = passed_error.stack + + Raygun.send raygun_error, custom_data + + + ############################################################################### + # Analytics + ############################################################################### + + ### + Reports call events for call tracking to Localytics. + @param event_name [z.tracking.EventName] String for call event + @param call_et [z.calling.Call] Call entity + @param attributes [Object] Attributes for the event + ### + track_event: (event_name, call_et, attributes) -> + if call_et + attributes = + conversation_participants: call_et.conversation_et.number_of_participants() + conversation_participants_in_call: call_et.max_number_of_participants + conversation_type: if call_et.is_group() then 'group' else 'one_to_one' + + if call_et.is_remote_videod() + event_name = event_name.replace '_call', '_video_call' + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event_name, attributes + + # Track the call duration. + track_duration: (call_et) => + duration = Math.floor (Date.now() - call_et.timer_start) / 1000 + if not window.isNaN duration + @logger.log @logger.levels.INFO, "Call duration: #{duration} seconds.", call_et.duration_time() + + if duration <= 15 + duration_bucket = '0s-15s' + else if duration <= 30 + duration_bucket = '16s-30s' + else if duration <= 60 + duration_bucket = '31s-60s' + else if duration <= 3 * 60 + duration_bucket = '61s-3min' + else if duration <= 10 * 60 + duration_bucket = '3min-10min' + else if duration <= 60 * 60 + duration_bucket = '10min-1h' + else + duration_bucket = '1h-infinite' + + attributes = + conversation_participants: call_et.conversation_et.number_of_participants() + conversation_participants_in_call: call_et.max_number_of_participants + conversation_type: if call_et.is_group() then 'group' else 'one_to_one' + duration: duration_bucket + duration_sec: duration + reason: call_et.finished_reason + + event_name = z.tracking.EventName.CALLING.ENDED_CALL + if call_et.is_remote_videod() + event_name = event_name.replace '_call', '_video_call' + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event_name, attributes diff --git a/app/script/telemetry/calling/ConnectionStats.coffee b/app/script/telemetry/calling/ConnectionStats.coffee new file mode 100644 index 00000000000..30e7ff9cebd --- /dev/null +++ b/app/script/telemetry/calling/ConnectionStats.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +# Connection stats entity. +class z.telemetry.calling.ConnectionStats + # Construct a new connection stats report. + constructor: -> + @timestamp = Date.now() + @connected = undefined + + @audio = new z.telemetry.calling.AudioStreamStats @timestamp + @peer_connection = new z.telemetry.calling.StreamStats @timestamp + @video = new z.telemetry.calling.VideoStreamStats @timestamp diff --git a/app/script/telemetry/calling/FlowTelemetry.coffee b/app/script/telemetry/calling/FlowTelemetry.coffee new file mode 100644 index 00000000000..004f1d56aa3 --- /dev/null +++ b/app/script/telemetry/calling/FlowTelemetry.coffee @@ -0,0 +1,458 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +# Flow telemetry entity. +class z.telemetry.calling.FlowTelemetry + ### + Construct new flow telemetry entity. + + @param id [String] Flow ID + @param remote_user_id [String] Remote user ID + @param call_et [z.calling.Call] Call entity + @param timings [z.telemetry.calling.CallSetupTimings] Timings of call setup steps + ### + constructor: (id, @remote_user_id, @call_et, timings) -> + @logger = new z.util.Logger "z.telemetry.calling.FlowTelemetry (#{id})", z.config.LOGGER.OPTIONS + + @id = id + @is_answer = false + @peer_connection = undefined + + @timings = timings + @statistics = new z.telemetry.calling.ConnectionStats() + + @stats_poller = undefined + + + ############################################################################### + # External misc + ############################################################################### + + ### + Create flow status report. + @param passed_error [Error] Optional error to be added to report + @return [Object] Report + ### + create_report: (passed_error) => + report = + meta: + browser_name: z.util.Environment.browser.name + browser_version: z.util.Environment.browser.version + flow_id: @id + id: @call_et.id + is_answer: @is_answer + session_id: @call_et.session_id() + telemetry: + statistics: @get_statistics() + timings: @get_timings() + + if @peer_connection + report.rtc_peer_connection = + ice_connection_state: @peer_connection.iceConnectionState + ice_gathering_state: @peer_connection.iceGatheringState + signaling_state: @peer_connection.signalingState + + if @peer_connection.localDescription? + rtc_peer_connection = + local_SDP: @peer_connection.localDescription.sdp + local_SDP_type: @peer_connection.localDescription.type + $.extend report.rtc_peer_connection, rtc_peer_connection + + if @peer_connection.remoteDescription? + rtc_peer_connection = + remote_SDP: @peer_connection.remoteDescription.sdp + remote_SDP_type: @peer_connection.remoteDescription.type + $.extend report.rtc_peer_connection, rtc_peer_connection + + if passed_error + report.error = passed_error + + return report + + ### + Check stream for flowing bytes. + + @param media_type [z.calling.enum.MediaType] Media type of stream + @param timeout [Number] Time in milliseconds since the check was scheduled + @param attempt [Number] Attempt of stream check + ### + check_stream: (media_type, timeout, attempt = 1) => + stats = @statistics[media_type] + if stats + seconds = attempt * timeout / 1000 + if stats.bytes_received is 0 and stats.bytes_sent is 0 + @logger.log @logger.levels.ERROR, "No '#{media_type}' flowing in either direction after #{seconds} seconds" + else if stats.bytes_received is 0 + @logger.log @logger.levels.ERROR, "No incoming '#{media_type}' received after #{seconds} seconds" + else if stats.bytes_sent is 0 + @logger.log @logger.levels.ERROR, "No outgoing '#{media_type}' sent after #{seconds} seconds" + else + @logger.log @logger.levels.DEBUG, "Stream has '#{media_type}' flowing properly both ways" + else + if @is_answer + @logger.log @logger.levels.INFO, "Check stream statistics of type '#{media_type}' delayed as we created this flow" + else + window.setTimeout => + @check_stream media_type, timeout, attempt++ + , timeout + + ### + Schedule check of stream activity. + @param timeout [Number] Milliseconds from now to execute the check + ### + schedule_check: (timeout) -> + window.setTimeout => + @check_stream z.calling.enum.MediaType.AUDIO, timeout + @check_stream z.calling.enum.MediaType.VIDEO, timeout if @call_et.is_remote_videod() + , timeout + + ### + Set the PeerConnection on the telemetry. + @param [RTCPeerConnection] PeerConnection to be used for telemetry + ### + set_peer_connection: (peer_connection) => + @peer_connection = peer_connection + + ### + Update 'is_answer' status of flow. + @param is_answer [Boolean] Is the flow an answer + ### + update_is_answer: (is_answer) => + @is_answer = is_answer + @timings.is_answer = is_answer + + + ############################################################################### + # Statistics + ############################################################################### + + # Flow connected. + connected: => + @statistics.connected = Date.now() + + ### + Return the statistics object. + @return [z.telemetry.calling.stats.ConnectionStats] Flow statistics + ### + get_statistics: => + return @statistics + + # Update statics for the last time and then reset them and the polling interval. + reset_statistics: => + return new Promise (resolve, reject) => + @_update_statistics() + .then => + resolve @statistics + @statistics = {} + .catch (error) => + @logger.log @logger.levels.WARN, "Failed to update flow networks stats: #{error.message}" + reject error + window.clearInterval @stats_poller + @stats_poller = undefined + + ### + Start statistics polling. + @param ice_connection_state [RTCIceConnectionState] Current state of ICE connection + ### + start_statistics: (ice_connection_state) => + if not @stats_poller + # Track call stats + @time_step z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_CONNECTED + $.extend @statistics, new z.telemetry.calling.ConnectionStats() + @connected() + + # Report calling stats within specified interval + window.setTimeout => + @_update_statistics() + .then => @logger.log @logger.levels.INFO, 'Flow network stats updated for the first time', @statistics + .catch (error) => @logger.log @logger.levels.WARN, "Failed to update flow networks stats: #{error.message}" + , 50 + @stats_poller = window.setInterval => + @_update_statistics() + .then => @logger.log @logger.levels.OFF, 'Flow network stats updated', @statistics + .catch (error) => @logger.log @logger.levels.WARN, "Flow networks stats not updated: #{error.message}" + , 2000 + + if ice_connection_state is z.calling.rtc.ICEConnectionState.COMPLETED + @time_step z.telemetry.calling.CallSetupSteps.ICE_CONNECTION_COMPLETED + + ### + Get current statistics from PeerConnection. + @private + @return [Promise] Promise to be resolved when stats are returned + ### + _update_statistics: => + @peer_connection.getStats null + .then (rtc_statistics) => + @logger.log @logger.levels.OFF, 'Received new statistics report for flow', rtc_statistics + + updated_statistics = new z.telemetry.calling.ConnectionStats() + + for key, report of rtc_statistics + switch report.type + when z.calling.rtc.StatsType.CANDIDATE_PAIR + updated_statistics = @_update_from_candidate_pair report, rtc_statistics, updated_statistics + when z.calling.rtc.StatsType.GOOGLE_CANDIDATE_PAIR + updated_statistics = @_update_peer_connection_bytes report, updated_statistics + updated_statistics = @_update_from_google_candidate_pair report, rtc_statistics, updated_statistics + when z.calling.rtc.StatsType.INBOUND_RTP + updated_statistics = @_update_peer_connection_bytes report, updated_statistics + updated_statistics = @_update_from_inbound_rtp report, updated_statistics + when z.calling.rtc.StatsType.OUTBOUND_RTP + updated_statistics = @_update_peer_connection_bytes report, updated_statistics + updated_statistics = @_update_from_outbound_rtp report, updated_statistics + when z.calling.rtc.StatsType.SSRC + updated_statistics = @_update_from_ssrc report, updated_statistics + + _calc_rate = (key, timestamp, type) => + bytes = (updated_statistics[key][type] - @statistics[key][type]) + time_span = (updated_statistics.timestamp - timestamp) + return window.parseInt 1000.0 * bytes / time_span, 10 + + # Calculate bit rate since last update + for key, value of updated_statistics + if _.isObject value + updated_statistics[key].bit_rate_mean_received = _calc_rate key, @statistics.connected, 'bytes_received' + updated_statistics[key].bit_rate_mean_sent = _calc_rate key, @statistics.connected, 'bytes_sent' + updated_statistics[key].bit_rate_current_received = _calc_rate key, @statistics.timestamp, 'bytes_received' + updated_statistics[key].bit_rate_current_sent = _calc_rate key, @statistics.timestamp, 'bytes_sent' + + $.extend @statistics, updated_statistics + @logger.log @logger.levels.OFF, 'Update of network stats for flow successful', @statistics + .catch (error) => + @logger.log @logger.levels.WARN, 'Update of network stats for flow failed', error + + ### + Update from z.calling.rtc.StatsType.CANDIDATE_PAIR report. + + @param report [Object] z.calling.rtc.StatsType.CANDIDATE_PAIR report + @param stats_reports [RTCStatsReport] Statistics report from PeerConnection + @param updated_stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_from_candidate_pair: (report, stat_reports, updated_stats) -> + if report.selected + updated_stats.peer_connection.local_candidate_type = stat_reports[report.localCandidateId].candidateType + updated_stats.peer_connection.remote_candidate_type = stat_reports[report.remoteCandidateId].candidateType + return updated_stats + + ### + Update from z.calling.rtc.StatsType.GOOGLE_CANDIDATE_PAIR report. + + @param report [Object] z.calling.rtc.StatsType.GOOGLE_CANDIDATE_PAIR report + @param stats_reports [RTCStatsReport] Statistics report from PeerConnection + @param updated_stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_from_google_candidate_pair: (report, stat_reports, updated_stats) -> + if report.googActiveConnection is 'true' + updated_stats.peer_connection.round_trip_time = window.parseInt report.googRtt, 10 + updated_stats.peer_connection.local_candidate_type = stat_reports[report.localCandidateId].candidateType + updated_stats.peer_connection.remote_candidate_type = stat_reports[report.remoteCandidateId].candidateType + return updated_stats + + ### + Update from z.calling.rtc.StatsType.INBOUND_RTP report. + + @param report [Object] z.calling.rtc.StatsType.INBOUND_RTP report + @param stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_from_inbound_rtp: (report, stats) -> + if report.mediaType in [z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaType.VIDEO] + stats[report.mediaType].bytes_received += report.bytesReceived if report.bytesReceived + stats[report.mediaType].frame_rate_received = window.parseInt report.framerateMean, 10 if report.framerateMean + return stats + + ### + Update from z.calling.rtc.StatsType.OUTBOUND_RTP report. + + @param report [Object] z.calling.rtc.StatsType.OUTBOUND_RTP report + @param stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_from_outbound_rtp: (report, stats) -> + if report.mediaType in [z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaType.VIDEO] + stats[report.mediaType].bytes_sent += report.bytesSent if report.bytesSent + stats[report.mediaType].frame_rate_sent = window.parseInt report.framerateMean, 10 if report.framerateMean + return stats + + ### + Update from statistics report. + + @param report [Object] Statistics report + @param stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_peer_connection_bytes: (report, stats) -> + stats.peer_connection.bytes_received += window.parseInt report.bytesReceived, 10 if report.bytesReceived + stats.peer_connection.bytes_sent += window.parseInt report.bytesSent, 10 if report.bytesSent + return stats + + ### + Update from z.calling.rtc.StatsType.SSRC report. + + @param report [Object] z.calling.rtc.StatsType.SSRC report + @param stats [z.telemetry.calling.ConnectionStats] Parsed flow statistics + @return [z.telemetry.calling.ConnectionStats] updated_stats + ### + _update_from_ssrc: (report, stats) => + if report.codecImplementationName + codec = "#{report.googCodecName} #{report.codecImplementationName}" + else + codec = report.googCodecName + + if report.audioOutputLevel + stream_stats = stats.audio + stream_stats.volume_received = window.parseInt report.audioOutputLevel, 10 + stream_stats.codec_received = codec + else if report.audioInputLevel + stream_stats = stats.audio + stream_stats.volume_sent = window.parseInt report.audioInputLevel, 10 + stream_stats.codec_sent = codec + else if @call_et.is_remote_videod() + stream_stats = stats.video + if report.googFrameHeightReceived + stream_stats.frame_height_received = window.parseInt report.googFrameHeightReceived, 10 + stream_stats.frame_rate_received = window.parseInt report.googFrameRateReceived, 10 + stream_stats.frame_width_received = window.parseInt report.googFrameWidthReceived, 10 + stream_stats.codec_received = codec + else if report.googFrameHeightSent + stream_stats.frame_height_sent = window.parseInt report.googFrameHeightSent, 10 + stream_stats.frame_rate_sent = window.parseInt report.googFrameRateSent, 10 if report.googFrameRateSent + stream_stats.frame_width_sent = window.parseInt report.googFrameWidthSent, 10 if report.googFrameWidthSent + stream_stats.codec_sent = codec + + if stream_stats + stream_stats.bytes_received += window.parseInt report.bytesReceived, 10 if report.bytesReceived + stream_stats.bytes_received = stats.peer_connection.bytes_received if stream_stats.bytes_received is 0 + stream_stats.bytes_sent += window.parseInt report.bytesSent, 10 if report.bytesSent + stream_stats.bytes_sent = stats.peer_connection.bytes_sent if stream_stats.bytes_sent is 0 + stream_stats.delay = window.parseInt report.googCurrentDelayMs, 10 if report.googCurrentDelayMs + stream_stats.round_trip_time = window.parseInt report.googRtt, 10 if report.googRtt + + return stats + + + ############################################################################### + # Timings + ############################################################################### + + ### + Return the step timings object. + @return [z.telemetry.calling.CallSetupTimings] Flow statistics + ### + get_timings: => + return @timings.get() + + time_step: (step) => + @timings.time_step step + + + ############################################################################### + # Reporting & Logging + ############################################################################### + + # Get full report. + get_report: => + return {} = + report: @create_report() + + # Log the flow to the browser console. + log_status: (participant_et) => + @logger.force_log "-- ID: #{@id}" + + if @remote_user isnt undefined + @logger.force_log "-- Remote user: #{participant_et.user.name()} (#{participant_et.user.id})" + + @logger.force_log "-- User is connected: #{participant_et.is_connected()}" + @logger.force_log "-- Flow is answer: #{@is_answer}" + + if @peer_connection + @logger.force_log "-- ICE connection: #{@peer_connection.iceConnectionState}" + @logger.force_log "-- ICE gathering: #{@peer_connection.iceGatheringState}" + + statistics = @get_statistics() + if statistics + # @note Types are 'none' if we cannot connect to the user (0 bytes flow) + @logger.force_log 'PeerConnection network statistics', statistics + @logger.force_log "-- Remote ICE candidate type: #{statistics.peer_connection.remote_candidate_type}" + @logger.force_log "-- Local ICE candidate type: #{statistics.peer_connection.local_candidate_type}" + # PeerConnection Stats + for key, value of statistics + if _.isObject value + @logger.force_log "Statistics for '#{key}':" + @logger.force_log "-- Bit rate received: #{value.bit_rate_received}" + @logger.force_log "-- Bit rate sent: #{value.bit_rate_sent}" + @logger.force_log "-- Bytes sent: #{value.bytes_sent}" + @logger.force_log "-- Bytes received: #{value.bytes_received}" + @logger.force_log "-- Rtt: #{value.rtt}" + media_types = [z.calling.enum.MediaType.AUDIO, z.calling.enum.MediaType.VIDEO] + if z.util.Environment.browser.chrome and key in media_types + @logger.force_log "-- Codec received: #{value.codec_received}" + @logger.force_log "-- Codec sent: #{value.codec_sent}" + @logger.force_log "-- Delay in ms: #{value.delay}" + if key is z.calling.enum.MediaType.VIDEO + @logger.force_log "-- Frame rate received: #{value.frame_rate_received}" + @logger.force_log "-- Frame rate sent: #{value.frame_rate_sent}" + continue if not z.util.Environment.browser.chrome + received_resolution = "#{value.frame_width_received}x#{value.frame_height_received}" + sent_resolution = "#{value.frame_width_sent}x#{value.frame_height_sent}" + @logger.force_log "-- Frame resolution received: #{received_resolution}" + @logger.force_log "-- Frame resolution sent: #{sent_resolution}" + else if key is z.calling.enum.MediaType.AUDIO + @logger.force_log "-- Volume received: #{value.volume_received}" + @logger.force_log "-- Volume sent: #{value.volume_sent}" + + log_timings: => + @timings.log() + + ### + Report an error to Raygun. + @param description [String] Error description + @param passed_error [Object] Error passed into the report + @param payload [Object] Additional payload for the custom data + ### + report_error: (description, passed_error, payload) => + custom_data = @create_report() + raygun_error = new Error description + + if passed_error + custom_data.error = passed_error + raygun_error.stack = passed_error.stack + + if payload + custom_data.payload = payload + + @logger.log @logger.levels.ERROR, description, custom_data + Raygun.send raygun_error, custom_data + + report_status: => + custom_data = @create_report() + @logger.log @logger.levels.INFO, 'Created flow status for call failure report', custom_data + return custom_data + + report_timings: => + custom_data = @timings.log() + Raygun.send new Error('Call setup step timings'), custom_data + @logger.log @logger.levels.INFO, + "Reported setup step timings of flow id '#{@id}' for call analysis", custom_data diff --git a/app/script/telemetry/calling/MediaStreamStats.coffee b/app/script/telemetry/calling/MediaStreamStats.coffee new file mode 100644 index 00000000000..ed4e70556c5 --- /dev/null +++ b/app/script/telemetry/calling/MediaStreamStats.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +class z.telemetry.calling.MediaStreamStats extends z.telemetry.calling.Stats + constructor: (timestamp) -> + super timestamp + @bit_rate_current_received = 0 + @bit_rate_current_sent = 0 + @bit_rate_mean_received = 0 + @bit_rate_mean_sent = 0 + @codec_received = '' + @codec_sent = '' + @delay = 0 diff --git a/app/script/telemetry/calling/Stats.coffee b/app/script/telemetry/calling/Stats.coffee new file mode 100644 index 00000000000..abd09dffbab --- /dev/null +++ b/app/script/telemetry/calling/Stats.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +# http://w3c.github.io/webrtc-stats/ +class z.telemetry.calling.Stats + constructor: (@timestamp) -> + @bytes_received = 0 + @bytes_sent = 0 + @packets_lost = 0 + @round_trip_time = 0 diff --git a/app/script/telemetry/calling/StreamStats.coffee b/app/script/telemetry/calling/StreamStats.coffee new file mode 100644 index 00000000000..d930072fb70 --- /dev/null +++ b/app/script/telemetry/calling/StreamStats.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +class z.telemetry.calling.StreamStats extends z.telemetry.calling.Stats + constructor: (timestamp) -> + super timestamp + @local_candidate_type = '' + @remote_candidate_type = '' diff --git a/app/script/telemetry/calling/VideoStreamStats.coffee b/app/script/telemetry/calling/VideoStreamStats.coffee new file mode 100644 index 00000000000..fdfba6dfe66 --- /dev/null +++ b/app/script/telemetry/calling/VideoStreamStats.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.telemetry ?= {} +z.telemetry.calling ?= {} + +class z.telemetry.calling.VideoStreamStats extends z.telemetry.calling.MediaStreamStats + constructor: (timestamp) -> + super timestamp + @media_type = z.calling.enum.MediaType.VIDEO + @frame_height_received = 0 + @frame_height_sent = 0 + @frame_rate_received = 0 + @frame_rate_sent = 0 + @frame_width_received = 0 + @frame_width_sent = 0 diff --git a/app/script/tracking/EventName.coffee b/app/script/tracking/EventName.coffee new file mode 100644 index 00000000000..33a47773fa5 --- /dev/null +++ b/app/script/tracking/EventName.coffee @@ -0,0 +1,95 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.tracking ?= {} + +### +Definition of events used for user-tracking with Localytics. +@note Event names should be descriptive (!) like "Item Purchased" instead of being programmatic like "itemPurchased". +### +z.tracking.EventName = + ACCOUNT: + LOGGED_IN: 'account.logged_in' + OPENED_LOGIN: 'account.opened_login' + APP_LAUNCH: 'appLaunch' + CALLING: + ENDED_CALL: 'calling.ended_call' + ESTABLISHED_CALL: 'calling.established_successful_call' + FAILED_REQUEST: 'calling.failed_request' + FAILED_REQUESTING_MEDIA: 'calling.failed_requesting_media' + FAILED_RTC: 'calling.failed_rtc' + INITIATED_CALL: 'calling.initiated_call' + MINIMIZED_FROM_FULLSCREEN: 'calling.minimized_from_fullscreen' + JOINED_CALL: 'calling.joined_call' + RECEIVED_CALL: 'calling.received_call' + SHARED_SCREEN: 'calling.shared_screen' + CONVERSATION: + ADD_TO_GROUP_CONVERSATION: 'addContactToGroupConversation' + CHARACTER_LIMIT_REACHED: 'conversation.character_limit_reached' + CREATE_GROUP_CONVERSATION: 'createGroupConversation' + DELETED_MESSAGE: 'conversation.deleted_message' + SELECTED_MESSAGE: 'conversation.selected_message' + MEDIA: + COMPLETED_MEDIA_ACTION: 'media.completed_media_action' + PLAYED_AUDIO_MESSAGE: 'media.played_audio_message' + PLAYED_VIDEO_MESSAGE: 'media.played_video_message' + E2EE: + CANNOT_DECRYPT_MESSAGE: 'e2ee.cannot_decrypt_message' + FILE: + UPLOAD_CANCELLED: 'file.cancelled_file_upload' + UPLOAD_INITIATED: 'file.initiated_file_upload' + UPLOAD_FAILED: 'file.failed_file_upload' + UPLOAD_SUCCESSFUL: 'file.successfully_uploaded_file' + UPLOAD_TOO_BIG: 'file.attempted_too_big_file_upload' + DOWNLOAD_INITIATED: 'file.initiated_file_download' + DOWNLOAD_FAILED: 'file.failed_file_download' + DOWNLOAD_SUCCESSFUL: 'file.successfully_downloaded_file' + IMAGE_SENT_ERROR: 'Image Sent Error' + NAVIGATION: + OPENED_WIRE_WEBSITE: 'navigation.opened_wire_website' + OPENED_TERMS: 'navigation.opened_terms' + ONBOARDING: + ADDED_PHOTO: 'onboarding.added_photo' + IMPORTED_CONTACTS: 'onboarding.imported_contacts' + PASSWORD_RESET: 'resetPassword' + PERMISSION: + ALLOW_NOTIFICATIONS: 'allowNotifications' + PROFILE_PICTURE_CHANGED: 'changedProfilePicture' + REGISTRATION: + ENTERED_CREDENTIALS: 'registration.entered_credentials' + OPENED_EMAIL_SIGN_UP: 'registration.opened_email_signup' + RESENT_EMAIL_VERIFICATION: 'registration.resent_email_verification' + SUCCEEDED: 'registration.succeeded' + SETTINGS: + IMPORTED_CONTACTS: 'settings.imported_contacts' + VIEWED_DEVICE: 'settings.viewed_device' + REMOVED_DEVICE: 'settings.removed_device' + SETTINGS_MENU: + SHOW_ABOUT_SCREEN: 'about' + SHOW_SUPPORT_PAGE: 'help' + SOUND_SETTINGS_CHANGED: 'soundIntensityPreference' + TELEMETRY: + APP_INITIALIZATION: 'telemetry.app_initialization' + TRACKING: + OPT_IN: 'Opt-in' + OPT_OUT: 'Opt-out' + UPLOADED_CONTACTS: 'uploadedContacts' # "source": "Gmail" + ANNOUNCE: + SENT: 'announce.sent' + CLICKED: 'announce.clicked' diff --git a/app/script/tracking/EventTrackingRepository.coffee b/app/script/tracking/EventTrackingRepository.coffee new file mode 100644 index 00000000000..566b7d01a9d --- /dev/null +++ b/app/script/tracking/EventTrackingRepository.coffee @@ -0,0 +1,266 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.tracking ?= {} + +z.tracking.config = + SESSION_TIMEOUT: 180000 # milliseconds + +LOCALYTICS = + APP_KEY: '905792736c9f17c3464fd4e-60d90c82-d14a-11e4-af66-009c5fda0a25' + TRACKING_INTERVAL: 60000 # milliseconds + DISABLED_DOMAINS: [ + 'localhost' + 'wire-webapp' + ] + +RAYGUN = + API_KEY: '5hvAMmz8wTXaHBYqu2TFUQ==' + +if z.util.Environment.frontend.is_production() + RAYGUN.API_KEY = 'lAkLCPLx3ysnsXktajeHmw==' + LOCALYTICS.APP_KEY = 'b929419faf17d843c16649c-f5cc4c44-ccb3-11e4-2efd-004a77f8b47f' + if z.util.Environment.electron + if z.util.Environment.os.mac + LOCALYTICS.APP_KEY = 'ad0b57c3c46d92daea395e0-146bc33e-6100-11e5-09e1-00deb82fd81f' + else if z.util.Environment.os.win + LOCALYTICS.APP_KEY = 'dfb424ad373e18163f25bc6-aa01a13c-6100-11e5-fb56-008b20abc1fa' + +### +Tracker for user actions which uses Localytics as a reference implementation but can be easily used with other services. + +@see https://support.localytics.com/Javascript +@see http://docs.localytics.com/#Dev/Instrument/js-tag-events.html +### +class z.tracking.EventTrackingRepository + constructor: (@user_repository, @conversation_repository) -> + @logger = new z.util.Logger 'z.tracking.EventTrackingRepository', z.config.LOGGER.OPTIONS + + @localytics = undefined # Localytics + @interval = undefined # Interval to track the Localytics session + @tracking_id = undefined # Tracking ID of the self user + @user_properties = undefined # Reference to the current user properties / settings + + @reported_errors = ko.observableArray() + @reported_errors.subscribe => @reported_errors [] if @reported_errors().length > 999 + @session_values = {} + + if @user_repository is undefined and @conversation_repository is undefined + @_start_session_without_user_tracking() + @_enable_error_reporting() + else + @session_started = Date.now() + @_reset_session_values() + + @_subscribe() + + _subscribe: -> + amplify.subscribe z.event.WebApp.ANALYTICS.INIT, @init + + ### + @param user_properties [z.user.UserProperties] + @param user_et [z.entity.User] + ### + init: (user_properties, user_et) => + @logger.log @logger.levels.INFO, 'Initialize tracking and error reporting' + @user_properties = user_properties + @tracking_id = user_et.tracking_id + @_start_session() + @_enable_error_reporting() if @_has_permission() + @_subscribe_to_events() + + ### + @see http://docs.localytics.com/#Dev/Integrate/web-options.html + ### + _init_localytics: (window, document, node_type, @localytics, c, script_node) => + options = + appVersion: z.util.Environment.version() + sessionTimeout: z.tracking.config.SESSION_TIMEOUT / 1000 + + @localytics = -> + (@localytics.q ?= []).push arguments + @localytics.t = new Date() + window.ll = @localytics + window['LocalyticsGlobal'] = 'll' + script_node = document.createElement node_type + script_node.src = 'https://web.localytics.com/v3/localytics.min.js' + (c = document.getElementsByTagName(node_type)[0]).parentNode.insertBefore script_node, c + @localytics 'init', LOCALYTICS.APP_KEY, options + @logger.log @logger.levels.INFO, 'Enabling Localytics reporting' + + _localytics_disabled: -> + if not z.util.get_url_parameter z.auth.URLParameter.LOCALYTICS + for domain in LOCALYTICS.DISABLED_DOMAINS when z.util.contains window.location.hostname, domain + @logger.log @logger.levels.WARN, 'Localytics reporting is disabled' + return true + return false + + _subscribe_to_events: -> + amplify.subscribe z.event.WebApp.ANALYTICS.EVENT, @_track_event + amplify.subscribe z.event.WebApp.ANALYTICS.SESSION.START, @_start_session + amplify.subscribe z.event.WebApp.ANALYTICS.SESSION.CLOSE, @_close_session + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATE.SEND_DATA, @_send_data + + _start_session_without_user_tracking: => + return if @_localytics_disabled() + @_init_localytics window, document, 'script', @localytics + @localytics 'open' + @localytics 'upload' + amplify.subscribe z.event.WebApp.ANALYTICS.EVENT, @_track_event + + _start_session: => + return if @_localytics_disabled() + if @localytics is undefined and @_has_permission() + @_init_localytics window, document, 'script', @localytics + @logger.log @logger.levels.INFO, 'Starting new Localytics session' + @localytics 'setCustomerId', @tracking_id + @localytics 'open' + @localytics 'upload' + @interval = window.setInterval @_tag_and_upload_session, LOCALYTICS.TRACKING_INTERVAL + + ### + @return [Boolean] true when "improve_wire" is set to "true". + ### + _has_permission: -> + return false if @user_properties is undefined + return false if @tracking_id is undefined + return @user_properties.settings.privacy.improve_wire + + _track_event: (event_name, attributes) => + if @session_values[event_name] isnt undefined + if attributes is undefined + # Increment session event value + @session_values[event_name] += 1 + else + if window.Number.isInteger attributes + @session_values[event_name] += attributes + else + @session_values[event_name] = attributes + else + # Logging events which are not bound to a session + @_tag_and_upload_event event_name, attributes + + _send_data: (send_data) => + if send_data is false + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.TRACKING.OPT_OUT + @_close_session() + @_disable_error_reporting() + else if @session is undefined + @_enable_error_reporting() + @_start_session() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.TRACKING.OPT_IN + + # @note We need a fat arrow here, because this function is executed from "window.setInterval" + # + _tag_and_upload_session: => + if @_has_permission() + number_of_connections = @user_repository.get_number_of_connections() + + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_CONTACTS] = @user_repository.connections().length + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_GROUP_CONVERSATIONS] = @conversation_repository.get_number_of_group_conversations() + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_INCOMING_CONNECTION_REQUESTS] = number_of_connections.incoming + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_OUTGOING_CONNECTION_REQUESTS] = number_of_connections.outgoing + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_ARCHIVED_CONVERSATIONS] = @conversation_repository.conversations_archived().length + @session_values[z.tracking.SessionEventName.INTEGER.TOTAL_SILENCED_CONVERSATIONS] = @conversation_repository.get_number_of_silenced_conversations() + + # Sanitize logging data + for key of @session_values + if key is 'undefined' + delete @session_values[key] + + # Log data + @logger.log @logger.levels.INFO, 'Uploading data...', @session_values + @_tag_and_upload_event 'session', @session_values + + _tag_and_upload_event: (event_name, attributes) => + return if @localytics is undefined + + @localytics 'tagEvent', event_name, attributes + @localytics 'upload' + + _close_session: => + return if @localytics is undefined + @logger.log @logger.levels.INFO, 'Closing Localytics session' + window.clearInterval @interval + + session_ended = Date.now() + session_duration = session_ended - @session_started + @session_values['sessionDuration'] = session_duration / 1000 + + @localytics 'upload' + @localytics 'close' + window.ll = undefined + @localytics = undefined + @_reset_session_values() + + _disable_error_reporting: -> + @logger.log @logger.levels.INFO, 'Disabling Raygun error reporting' + Raygun.detach() + + _enable_error_reporting: -> + @logger.log @logger.levels.INFO, 'Enabling Raygun error reporting' + options = + ignoreAjaxAbort: true + ignoreAjaxError: true + excludedHostnames: [ + 'localhost' + 'wire.ms' + ] + ignore3rdPartyErrors: true + + options.debugMode = not z.util.Environment.frontend.is_production() + + Raygun.init(RAYGUN.API_KEY, options).attach() + ### + Adding a version to the Raygun reports to identify which version of the Wire ran into the issue. + + @note We cannot use our own version string as it has to be in a certain format + @see https://github.com/MindscapeHQ/raygun4js#version-filtering + ### + Raygun.setUser @tracking_id + Raygun.setVersion z.util.Environment.version false if not z.util.Environment.frontend.is_localhost() + Raygun.withCustomData {electron_version: z.util.Environment.version true} if z.util.Environment.electron + Raygun.onBeforeSend @_check_error_payload + + ### + Checks if a Raygun payload has been already reported. + + @see https://github.com/MindscapeHQ/raygun4js#onbeforesend + @param [JSON] raygun_payload + @return [JSON|Boolean] Returns the original payload if it is an unreported error, otherwise "false". + ### + _check_error_payload: (raygun_payload) => + error_hash = objectHash.sha1 raygun_payload.Details.Error + if @reported_errors().includes error_hash + return false + else + @reported_errors.push error_hash + return raygun_payload + + _reset_session_values: => + for index, event_name of z.tracking.SessionEventName.INTEGER + @session_values[event_name] = 0 + + for index, event_name of z.tracking.SessionEventName.BOOLEAN + @session_values[event_name] = false + + print_session_values: -> + for key, value of @session_values + @logger.force_log "¬ #{key}: #{value}" + return undefined diff --git a/app/script/tracking/SessionEventName.coffee b/app/script/tracking/SessionEventName.coffee new file mode 100644 index 00000000000..58a336e05ca --- /dev/null +++ b/app/script/tracking/SessionEventName.coffee @@ -0,0 +1,49 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.tracking ?= {} + +z.tracking.SessionEventName = + BOOLEAN: + SEARCHED_FOR_PEOPLE: 'searchedForPeople' + INTEGER: + CONNECT_REQUEST_ACCEPTED: 'connectRequestsAcceptedActual' + CONNECT_REQUEST_SENT: 'connectRequestsSentActual' + CONVERSATION_RENAMED: 'conversationRenamesActual' + EVENT_HIDDEN_DUE_TO_DUPLICATE_ID: 'eventHiddenDueToDuplicateIDActual' + EVENT_HIDDEN_DUE_TO_DUPLICATE_NONCE: 'eventHiddenDueToDuplicateNonceActual' + IMAGE_DETAIL_VIEW_OPENED: 'imageContentsClicksActual' + IMAGE_SENT: 'imagesSentActual' + INCOMING_CALL_ACCEPTED: 'incomingCallsAcceptedActual' + INCOMING_CALL_MUTED: 'incomingCallsMutedActual' + MESSAGE_SENT: 'textMessagesSentActual' # No differentiation between message with text or image + PING_SENT: 'pingsSentActual' # No differentiation between ping or hot ping + SEARCH_OPENED: 'openedSearchActual' + SOUNDCLOUD_CONTENT_CLICKED: 'soundcloudContentClicksActual' + SOUNDCLOUD_LINKS_SENT: 'soundcloudLinksSentActual' + TOTAL_ARCHIVED_CONVERSATIONS: 'totalArchivedConversationsActual' + TOTAL_CONTACTS: 'totalContactsActual' + TOTAL_GROUP_CONVERSATIONS: 'totalGroupConversationsActual' + TOTAL_INCOMING_CONNECTION_REQUESTS: 'totalIncomingConnectionRequestsActual' + TOTAL_OUTGOING_CONNECTION_REQUESTS: 'totalOutgoingConnectionRequestsActual' + TOTAL_SILENCED_CONVERSATIONS: 'totalSilencedConversationsActual' + USERS_ADDED_TO_CONVERSATIONS: 'usersAddedToConversationsActual' + VOICE_CALL_INITIATED: 'voiceCallsInitiatedActual' + YOUTUBE_CONTENT_CLICKED: 'youtubeContentClicksActual' + YOUTUBE_LINKS_SENT: 'youtubeLinksSentActual' diff --git a/app/script/tracking/event/PhoneVerification.coffee b/app/script/tracking/event/PhoneVerification.coffee new file mode 100644 index 00000000000..76450fc5071 --- /dev/null +++ b/app/script/tracking/event/PhoneVerification.coffee @@ -0,0 +1,36 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.tracking ?= {} +z.tracking.event ?= {} + +class z.tracking.event.PhoneVerification + ### + Construct a phone verification event. + + @param context [String] <"registration"|"postLogin"|"signIn"> + @param state [String] <"succeeded"|"error"|"resent"> + @param description [String] <"codeRequestError"|undefined> + ### + constructor: (@context, @state, @description) -> + @name = 'PhoneVerification' + @attributes = + context: @context + state: @state + description: @description diff --git a/app/script/tracking/event/PictureTakenEvent.coffee b/app/script/tracking/event/PictureTakenEvent.coffee new file mode 100644 index 00000000000..33186468fbf --- /dev/null +++ b/app/script/tracking/event/PictureTakenEvent.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.tracking ?= {} +z.tracking.event ?= {} + +class z.tracking.event.PictureTakenEvent + # Construct a phone verification event. + # + # @param [String] context + # @param [String] source + # @param [String] trigger + # + constructor: (@context, @source, @trigger) -> + @name = 'PictureTaken' + @attributes = + context: @context + source: @source + trigger: @trigger diff --git a/app/script/ui/Shortcut.coffee b/app/script/ui/Shortcut.coffee new file mode 100644 index 00000000000..88c3d6f9df6 --- /dev/null +++ b/app/script/ui/Shortcut.coffee @@ -0,0 +1,218 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ui ?= {} + +z.ui.ShortcutType = + ADD_PEOPLE: 'add_people' + ARCHIVE: 'archive' + CALL_IGNORE: 'ignore' + CALL_MUTE: 'mute_call' + DEBUG: 'debug' + NEXT: 'next' + PEOPLE: 'people' + PING: 'ping' + PREV: 'prev' + SILENCE: 'silence' + START: 'start' + +z.ui.Shortcut = do -> + + shortcut_map = {} + + shortcut_map[z.ui.ShortcutType.ADD_PEOPLE] = + shortcut: + webapp: + osx: 'command + shift + k' + pc: 'ctrl + shift + k' + electron: + osx: 'command + shift + k' + pc: 'ctrl + shift + k' + menu: true + event: z.event.WebApp.SHORTCUT.ADD_PEOPLE + + shortcut_map[z.ui.ShortcutType.ARCHIVE] = + shortcut: + webapp: + osx: 'command + alt + shift + d' + pc: 'ctrl + alt + d' + electron: + osx: 'command + d' + pc: 'ctrl + d' + menu: true + event: z.event.WebApp.SHORTCUT.ARCHIVE + + shortcut_map[z.ui.ShortcutType.CALL_IGNORE] = + shortcut: + webapp: + osx: 'command + alt + .' + pc: 'ctrl + alt + .' + electron: + osx: 'command + .' + pc: 'ctrl + .' + event: z.event.WebApp.SHORTCUT.CALL_IGNORE + + shortcut_map[z.ui.ShortcutType.CALL_MUTE] = + shortcut: + webapp: + osx: 'command + alt + m' + pc: 'ctrl + alt + m' + electron: + osx: 'command + alt + m' + pc: 'ctrl + alt + m' + event: z.event.WebApp.SHORTCUT.CALL_MUTE + + shortcut_map[z.ui.ShortcutType.PREV] = + shortcut: + webapp: + osx: 'command + alt + down' + pc: 'alt + shift + down' + electron: + osx: 'command + alt + down' + pc: 'alt + shift + down' + menu: true + event: z.event.WebApp.SHORTCUT.PREV + + shortcut_map[z.ui.ShortcutType.NEXT] = + shortcut: + webapp: + osx: 'command + alt + up' + pc: 'alt + shift + up' + electron: + osx: 'command + alt + up' + pc: 'alt + shift + up' + menu: true + event: z.event.WebApp.SHORTCUT.NEXT + + shortcut_map[z.ui.ShortcutType.PING] = + shortcut: + webapp: + osx: 'command + alt + k' + pc: 'ctrl + alt + k' + electron: + osx: 'command + k' + pc: 'ctrl + k' + menu: true + event: z.event.WebApp.SHORTCUT.PING + + shortcut_map[z.ui.ShortcutType.PEOPLE] = + shortcut: + webapp: + osx: 'command + alt + shift + i' + pc: 'ctrl + alt + i' + electron: + osx: 'command + i' + pc: 'ctrl + i' + menu: true + event: z.event.WebApp.SHORTCUT.PEOPLE + + shortcut_map[z.ui.ShortcutType.SILENCE] = + shortcut: + webapp: + osx: 'command + alt + s' + pc: 'ctrl + alt + s' + electron: + osx: 'command + alt + s' + pc: 'ctrl + alt + s' + menu: true + event: z.event.WebApp.SHORTCUT.SILENCE + + shortcut_map[z.ui.ShortcutType.START] = + shortcut: + webapp: + osx: 'command + alt + graveaccent' # KeyboardJS fires this when using cmd + alt + n + pc: 'ctrl + alt + graveaccent' + electron: + osx: 'command + n' + pc: 'ctrl + n' + menu: true + + event: z.event.WebApp.SHORTCUT.START + + if $('#debug').length isnt 0 + shortcut_map[z.ui.ShortcutType.DEBUG] = + shortcut: + webapp: + osx: 'command + alt + g' + pc: 'ctrl + alt + g' + electron: + osx: 'command + alt + g' + pc: 'ctrl + alt + g' + event: z.event.WebApp.SHORTCUT.DEBUG + + _register_event = (platform_specific_shortcut, event) -> + + # bind also 'command + alt + n' for start shortcut + if z.util.contains platform_specific_shortcut, 'graveaccent' + replaced_shortcut = platform_specific_shortcut.replace 'graveaccent', 'n' + _register_event replaced_shortcut, event + + keyboardJS.on platform_specific_shortcut, (e) -> + keyboardJS.releaseKey e.keyCode + + # hotfix WEBAPP-1916 + return if (z.util.contains(platform_specific_shortcut, 'command') and not e.metaKey) + + e.preventDefault() + amplify.publish event + + get_beautified_shortcut_mac = (shortcut) -> + return shortcut + .replace /\+/g, '' + .replace /\s+/g, '' + .replace 'alt', '⌥' + .replace 'command', '⌘' + .replace 'shift', '⇧' + .replace 'up', '↑' + .replace 'down', '↓' + .replace 'graveaccent', 'n' + .toUpperCase() + + get_beautified_shortcut_win = (shortcut) -> + return shortcut + .replace 'up', '↑' + .replace 'down', '↓' + .replace 'graveaccent', 'n' + .replace /\w+/g, (string) -> z.util.capitalize_first_char string + + get_shortcut = (shortcut_name) -> + platform = if z.util.Environment.electron then 'electron' else 'webapp' + platform_shortcuts = shortcut_map[shortcut_name].shortcut[platform] + os_shortcut = if z.util.Environment.os.mac then platform_shortcuts.osx else platform_shortcuts.pc + return os_shortcut + + get_shortcut_tooltip = (shortcut_name) -> + shortcut = get_shortcut shortcut_name + if shortcut + return get_beautified_shortcut_mac shortcut if z.util.Environment.os.mac + return get_beautified_shortcut_win shortcut + + _init = -> + for shortcut, data of shortcut_map + continue if z.util.Environment.electron and shortcut_map[shortcut].shortcut.electron.menu + _register_event get_shortcut(shortcut), data['event'] + + _init() + + return {} = + shortcut_map: shortcut_map + get_shortcut: get_shortcut + get_shortcut_tooltip: get_shortcut_tooltip + get_beautified_shortcut_mac: get_beautified_shortcut_mac + get_beautified_shortcut_win: get_beautified_shortcut_win diff --git a/app/script/ui/WindowHandler.coffee b/app/script/ui/WindowHandler.coffee new file mode 100644 index 00000000000..d74f83307fe --- /dev/null +++ b/app/script/ui/WindowHandler.coffee @@ -0,0 +1,92 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ui ?= {} + + +class z.ui.WindowHandler + constructor: -> + @logger = new z.util.Logger 'z.ui.WindowHandler', z.config.LOGGER.OPTIONS + + @width = 0 + @height = 0 + + @is_visible = true + @lost_focus_interval_time = (z.tracking.config.SESSION_TIMEOUT / 3) + @lost_focus_interval = undefined + @lost_focus_on = undefined + + return @ + + init: => + @width = $(window).width() + @height = $(window).height() + @_listen_to_window_resize() + @_listen_to_visibility_change => + if document.visibilityState is 'visible' + @logger.log 'Webapp is visible' + @is_visible = true + window.clearInterval @lost_focus_interval + @lost_focus_interval = undefined + @lost_focus_on = undefined + amplify.publish z.event.WebApp.ANALYTICS.SESSION.START + else + @logger.log 'Webapp is hidden' + @is_visible = false + if @lost_focus_interval is undefined + @lost_focus_on = Date.now() + @lost_focus_interval = window.setInterval (=> @_check_for_timeout()), @lost_focus_interval_time + return @ + + _listen_to_window_resize: => + $(window).on 'resize', => + current_height = $(window).height() + current_width = $(window).width() + + change_in_width = @width - current_width + change_in_height = @height - current_height + + amplify.publish z.event.WebApp.WINDOW.RESIZE.WIDTH, change_in_width + amplify.publish z.event.WebApp.WINDOW.RESIZE.HEIGHT, change_in_height + + @width = current_width + @height = current_height + + _listen_to_visibility_change: (callback) -> + property_hidden = undefined + property_visibility_change = undefined + + if typeof document.hidden isnt 'undefined' + property_hidden = 'hidden' + property_visibility_change = 'visibilitychange' + else if typeof document.msHidden isnt 'undefined' + property_hidden = 'msHidden' + property_visibility_change = 'msvisibilitychange' + else if typeof document.webkitHidden isnt 'undefined' + property_hidden = 'webkitHidden' + property_visibility_change = 'webkitvisibilitychange' + + if property_hidden + $(document).on property_visibility_change, -> + callback() + + _check_for_timeout: -> + in_background_since = Date.now() - @lost_focus_on + if in_background_since >= z.tracking.config.SESSION_TIMEOUT + amplify.publish z.event.WebApp.ANALYTICS.SESSION.CLOSE diff --git a/app/script/user/ConnectionLevel.coffee b/app/script/user/ConnectionLevel.coffee new file mode 100644 index 00000000000..45669721c1d --- /dev/null +++ b/app/script/user/ConnectionLevel.coffee @@ -0,0 +1,28 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +# Enum of different connection levels. +z.user.ConnectionLevel = + UNKNOWN: -1 + NO_CONNECTION: 0 + KNOWN_CONTACT: 1 + FRIEND_OF_FRIEND: 2 + FRIEND_OF_FRIEND_OF_FRIEND: 3 diff --git a/app/script/user/ConnectionStatus.coffee b/app/script/user/ConnectionStatus.coffee new file mode 100644 index 00000000000..548d22d4d50 --- /dev/null +++ b/app/script/user/ConnectionStatus.coffee @@ -0,0 +1,30 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +# Enum of different connection status. +z.user.ConnectionStatus = + ACCEPTED: 'accepted' + BLOCKED: 'blocked' + CANCELLED: 'cancelled' + IGNORED: 'ignored' + PENDING: 'pending' + SENT: 'sent' + UNKNOWN: '' diff --git a/app/script/user/UserConnectionMapper.coffee b/app/script/user/UserConnectionMapper.coffee new file mode 100644 index 00000000000..16d7db7bc1f --- /dev/null +++ b/app/script/user/UserConnectionMapper.coffee @@ -0,0 +1,64 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +# Connection mapper to convert all server side JSON connections into core connection entities. +class z.user.UserConnectionMapper + # Construct a new user mapper. + constructor: -> + @logger = new z.util.Logger 'z.user.UserConnectionMapper', z.config.LOGGER.OPTIONS + + ### + Converts JSON connection into connection entity. + + @param data [JSON] Connection data + + @return [z.entity.Connection] Mapped connection entity + ### + @map_user_connection_from_json = (data) -> + connection_et = new z.entity.Connection() + return @update_user_connection_from_json connection_et, data + + ### + Convert multiple JSON connections into connection entities. + + @param json [Object] Connection data + + @return [Array] Mapped connection entities + ### + @map_user_connections_from_json = (data) -> + return (@map_user_connection_from_json connection for connection in data when connection isnt undefined) + + ### + Maps JSON connection into a blank connection entity or updates an existing one. + + @param connection_et [z.entity.Connection] Connection entity that the info shall be mapped to + @param data [JSON] Connection data + + @return [z.entity.Connection] Mapped connection entity + ### + @update_user_connection_from_json = (connection_et, data) -> + connection_et.status data.status + connection_et.conversation_id = data.conversation + connection_et.to = data.to + connection_et.from = data.from + connection_et.last_update = data.last_update + connection_et.message = data.message + return connection_et diff --git a/app/script/user/UserError.coffee b/app/script/user/UserError.coffee new file mode 100644 index 00000000000..1b7454aeb2c --- /dev/null +++ b/app/script/user/UserError.coffee @@ -0,0 +1,35 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +class z.user.UserError + constructor: (message, type) -> + @name = @constructor.name + @message = message + @type = type + @stack = (new Error()).stack + + @:: = new Error() + @::constructor = @ + @::TYPE = { + NO_CLIENTS: 'z.user.UserError::TYPE.NO_CLIENTS' + PRE_KEY_NOT_FOUND: 'z.user.UserError::TYPE.PRE_KEY_NOT_FOUND' + REQUEST_FAILURE: 'z.user.UserError::TYPE.REQUEST_FAILURE' + } diff --git a/app/script/user/UserMapper.coffee b/app/script/user/UserMapper.coffee new file mode 100644 index 00000000000..fda9d3ca72c --- /dev/null +++ b/app/script/user/UserMapper.coffee @@ -0,0 +1,112 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +# User mapper to convert all server side JSON users into core user entities. +class z.user.UserMapper + ### + Construct a new User Mapper. + + @param asset_service [z.assets.AssetService] Backend REST API asset service implementation + ### + constructor: (@asset_service) -> + @logger = new z.util.Logger 'z.user.UserMapper', z.config.LOGGER.OPTIONS + + ### + Converts JSON user into user entity. + + @param data [Object] User data + + @return [z.entity.User] Mapped user entity + ### + map_user_from_object: (data) -> + user = new z.entity.User() + return @update_user_from_object user, data + + ### + Convert multiple JSON users into user entities. + + @note Return an empty array in any case to prevent crashes. + + @param json [Object] User data + + @return [Array] Mapped user entities + ### + map_users_from_object: (data) -> + if data? + return (@map_user_from_object user for user in data when user isnt undefined) + else + @logger.log @logger.levels.WARN, 'We got no user data from the backend' + return [] + + ### + Maps JSON user into a blank user entity or updates an existing one. + + @note Mapping of single properties to an existing user happens when the user changes his name or accent color. + + @param user_et [z.entity.User] User entity that the info shall be mapped to + @param data [Object] User data + + @return [z.entity.User] Mapped user entity + ### + update_user_from_object: (user_et, data) -> + return if not data? + # It's a new user + if data.id? and user_et.id is '' + user_et.id = data.id + user_et.joaat_hash = z.util.Crypto.Hashing.joaat_hash data.id + # We are trying to update non-matching users + else if user_et.id isnt '' and data.id isnt user_et.id + throw new Error('updating wrong user_et') + + if data.email? + user_et.email data.email + + if data.phone? + user_et.phone data.phone + + if data.name? + user_et.name data.name.trim() + + if data.accent_id? and data.accent_id isnt 0 + user_et.accent_id data.accent_id + + if data.picture? + user_et.raw_pictures data.picture + + if data.picture?[0]? + preview_picture = data.picture[0] + + if preview_picture.info.public isnt true + @logger.log @logger.levels.WARN, "User ID \"#{user_et.id}\" has a private profile picture." + + url = @asset_service.generate_asset_url preview_picture.id, user_et.id + user_et.picture_preview "url('#{url}')" + + if data.picture?[1]? + medium_picture = data.picture[1] + + if medium_picture.info.public isnt true + @logger.log @logger.levels.WARN, "User ID \"#{user_et.id}\" has a private medium picture." + + url = @asset_service.generate_asset_url medium_picture.id, user_et.id + user_et.picture_medium "url('#{url}')" + + return user_et diff --git a/app/script/user/UserProperties.coffee b/app/script/user/UserProperties.coffee new file mode 100644 index 00000000000..a0d8d9f66e9 --- /dev/null +++ b/app/script/user/UserProperties.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +class z.user.UserProperties + constructor: -> + @version = 1 + @settings = + call: + mute: false + permissions: + notifications: z.util.BrowserPermissionType.DEFAULT + privacy: + report_errors: true + improve_wire: true + sound: + alerts: z.audio.AudioSetting.ALL + @contact_import = + google: undefined + osx: undefined + @has_created_conversation = false + @enable_debugging = false diff --git a/app/script/user/UserRepository.coffee b/app/script/user/UserRepository.coffee new file mode 100644 index 00000000000..ae85980b053 --- /dev/null +++ b/app/script/user/UserRepository.coffee @@ -0,0 +1,694 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +# User repository for all user and connection interactions with the user service. +class z.user.UserRepository + ### + Construct a new User repository. + + @param user_service [z.user.UserService] Backend REST API user service implementation + @param asset_service [z.assets.AssetService] Backend REST API asset service implementation + @param search_service [z.search.SearchService] Backend REST API search service implementation + @param client_repository [z.client.ClientRepository] Repository for all client interactions + @param cryptography_repository [z.cryptography.CryptographyRepository] Repository for all cryptography interactions + ### + constructor: (@user_service, @asset_service, @search_service, @client_repository, @cryptography_repository) -> + @logger = new z.util.Logger 'z.user.UserRepository', z.config.LOGGER.OPTIONS + + @connection_mapper = new z.user.UserConnectionMapper() + @user_mapper = new z.user.UserMapper @asset_service + + @self = ko.observable() + @users = ko.observableArray [] + @connections = ko.observableArray [] + @properties = new z.user.UserProperties() + + @connect_requests = ko.computed => + user_ets = [] + for user_et in @users() + user_ets.push user_et if user_et.connection().status() is z.user.ConnectionStatus.PENDING + return user_ets + .extend rateLimit: 50 + + amplify.subscribe z.event.WebApp.PROPERTIES.CHANGE.DEBUG, @save_property_enable_debugging + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATED, @properties_updated + amplify.subscribe z.event.Backend.USER.CONNECTION, @user_connection + amplify.subscribe z.event.Backend.USER.UPDATE, @user_update + + + ############################################################################### + # Connections + ############################################################################### + + ### + Accept a connection request. + + @param user_et [z.entity.User] User to update connection with + @param show_conversation [Boolean] Show new conversation on success + @return [Promise] Promise that resolves when the connection request was accepted + ### + accept_connection_request: (user_et, show_conversation = false) => + @_update_connection_status user_et, z.user.ConnectionStatus.ACCEPTED, show_conversation + .then -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.CONNECT_REQUEST_ACCEPTED + + add_client_to_user: (user_id, client_et) => + return Promise.resolve() + .then => + user_et = @find_user user_id + @client_repository._save_client user_id, client_et.to_json() + .then -> + user_et.add_client client_et + return user_et + + ### + Block a user. + @param user_et [z.entity.User] User to block + @return [Promise] Promise that resolves when the user was blocked + ### + block_user: (user_et) => + @_update_connection_status user_et, z.user.ConnectionStatus.BLOCKED + + ### + Cancel a connection request. + @param user_et [z.entity.User] User to cancel the sent connection request + @param next_conversation_et [z.entity.Conversation] Optional conversation to be switched to + @return [Promise] Promise that resolves when an outgoing connection request was cancelled + ### + cancel_connection_request: (user_et, next_conversation_et) => + @_update_connection_status user_et, z.user.ConnectionStatus.CANCELLED + .then -> + amplify.publish z.event.WebApp.CONVERSATION.SHOW, next_conversation_et if next_conversation_et + + + ### + Create a connection request. + + @param user_et [z.entity.User] User to connect to + @param message [String] Connection message + @param show_conversation [Boolean] Should we open the new conversation + @return [Promise] Promise that resolves when the connection request was successfully created + ### + create_connection: (user_et, show_conversation = false) => + return Promise.resolve() + .then => + connect_message = z.localization.Localizer.get_text + id: z.string.connection_request_message + replace: [ + {placeholder: '%@.first_name', content: user_et.name()} + {placeholder: '%s.first_name', content: @self().name()} + ] + + @user_service.create_connection user_et.id, user_et.name(), connect_message + .then (response) => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.CONNECT_REQUEST_SENT + @user_connection response, show_conversation + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to send connection request to user '#{user_et.id}': #{error.message}", error + + ### + Create a connection request from generic invite token + @param token [String] + ### + create_connection_from_invite_token: (token) => + user_id = z.util.Invite.get_user_from_invitation_token token + + return if not user_id or @self().id is user_id + + @get_user_by_id user_id, (user_et) => + connection_et = user_et.connection() + return if connection_et.status() in [z.user.ConnectionStatus.BLOCKED, z.user.ConnectionStatus.ACCEPTED, + z.user.ConnectionStatus.SENT] + + if connection_et.status() is z.user.ConnectionStatus.PENDING + @accept_connection_request user_et + else + @create_connection user_et + + ### + Get a connection for a user ID. + @param user_id [String] User ID + @return [z.entity.Connection] User connection entity + ### + get_connection_by_user_id: (user_id) -> + for connection_et in @connections() + return connection_et if connection_et.to is user_id + + ### + Get a connection for a conversation ID. + @param conversation_id [String] Conversation ID + @return [z.entity.Connection] User connection entity + ### + get_connection_by_conversation_id: (conversation_id) -> + for connection_et in @connections() + return connection_et if connection_et.conversation_id is conversation_id + + ### + Get all user connections from backend and store users of that connection. + + @note Initially called by Wire for Web's app start to retrieve user entities and their connections. + + @param limit [Integer] Query limit for user connections + @param user_id [String] User ID of the latest connection + @param connection_ets [Array + return new Promise (resolve, reject) => + @user_service.get_own_connections limit, user_id + .then (response) => + if response.connections.length > 0 + new_connection_ets = @connection_mapper.map_user_connections_from_json response.connections + connection_ets = connection_ets.concat new_connection_ets + + if response.has_more + last_connection_et = connection_ets[connection_ets.length - 1] + @get_connections limit, last_connection_et.to, connection_ets + .then => resolve @connections() + else if connection_ets.length > 0 + @update_user_connections connection_ets + .then => resolve @connections() + else + resolve @connections() + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to retrieve connections from backend: #{error.message}", error + reject error + + ### + Ignore connection request. + @param user_et [z.entity.User] User to ignore the connection request + @return [Promise] Promise that resolves when an incoming connection request was ignored + ### + ignore_connection_request: (user_et) => + @_update_connection_status user_et, z.user.ConnectionStatus.IGNORED + + ### + Unblock a user. + + @param user_et [z.entity.User] User to unblock + @param show_conversation [Boolean] Show new conversation on success + @return [Promise] Promise that resolves when a user was unblocked + ### + unblock_user: (user_et, show_conversation = true) => + @_update_connection_status user_et, z.user.ConnectionStatus.ACCEPTED, show_conversation + + ### + Update the user connections and get the matching users. + @param connection_ets [Array] Connection entities + @return [Promise] Promise that resolves when all user connections have been updated + ### + update_user_connections: (connection_ets) => + return new Promise (resolve) => + z.util.ko_array_push_all @connections, connection_ets + + # Apply connection to other user entities (which are not us) + user_ids = (connection_et.to for connection_et in connection_ets) + + if user_ids.length > 0 + @get_users_by_id user_ids, (user_ets) => + @_assign_connection user_et for user_et in user_ets + @_assign_all_clients() + .then -> resolve() + + # Assign all locally stored clients to the users. + _assign_all_clients: => + @client_repository.get_all_clients_from_db() + .then (user_client_map) => + @logger.log "Found locally stored clients for '#{Object.keys(user_client_map).length}' users", user_client_map + user_ids = (user_id for user_id, client_ets of user_client_map) + @get_users_by_id user_ids, (user_ets) => + for user_et in user_ets + @logger.log "Found '#{user_client_map[user_et.id].length}' clients for '#{user_et.name()}'", user_client_map[user_et.id] + user_et.devices user_client_map[user_et.id] + + # Assign connections to the users. + _assign_connection: (user_et) => + connection_et = @get_connection_by_user_id user_et.id + user_et.connection connection_et if connection_et + + ### + Update the status of a connection. + + @private + @param user_et [z.entity.User] User to update connection with + @param status [String] Connection status + @param show_conversation [Boolean] Show conversation on success + @return [Promise] Promise that resolves when the connection status was updated + ### + _update_connection_status: (user_et, status, show_conversation = false) => + return Promise.resolve() + .then => + # check if the status has changed + return if user_et.connection().status() is status + return @user_service.update_connection_status user_et.id, status + .then (response) => + @user_connection response, show_conversation + .catch (error) => + @logger.log @logger.levels.ERROR, + "Connection status change to '#{status}' for user '#{user_et.id}' failed: #{error.message}", error + custom_data = + failed_action: status + server_error: error + Raygun.send new Error('Connection Status change failed'), custom_data + + ############################################################################### + # Events + ############################################################################### + + ### + Convert a JSON event into an entity and get the matching conversation. + @param event_json [Object] JSON data of 'user.connection' event + @param show_conversation [Boolean] Should the new conversation be opened? + ### + user_connection: (event_json, show_conversation) => + return if not event_json? + event_json = event_json.connection or event_json + + connection_et = @get_connection_by_user_id event_json.to + previous_status = null + + if connection_et? + previous_status = connection_et.status() + @connection_mapper.update_user_connection_from_json connection_et, event_json + else + connection_et = @connection_mapper.map_user_connection_from_json event_json + @update_user_connections [connection_et] + + if connection_et + if previous_status is z.user.ConnectionStatus.SENT and connection_et.status() is z.user.ConnectionStatus.ACCEPTED + @update_user_by_id connection_et.to + @_send_user_connection_notification connection_et, previous_status + amplify.publish z.event.WebApp.CONVERSATION.MAP_CONNECTION, [connection_et], show_conversation + + ### + Use a JSON event to update the matching user. + @param event_json [Object] JSON data + ### + user_update: (event_json) => + if event_json.user.id is @self().id + @user_mapper.update_user_from_object @self(), event_json.user + else + @get_user_by_id event_json.user.id, (user_et) => + @user_mapper.update_user_from_object user_et, event_json.user if user_et? + + ### + Send the user connection notification. + @param connection_et [z.entity.Connection] Connection entity + @param previous_status [z.user.ConnectionStatus] Previous connection status + ### + _send_user_connection_notification: (connection_et, previous_status) => + # We accepted the connection request or unblocked the user + no_notification = [z.user.ConnectionStatus.BLOCKED, z.user.ConnectionStatus.PENDING] + return if previous_status in no_notification and connection_et.status() is z.user.ConnectionStatus.ACCEPTED + + @get_user_by_id connection_et.to, (user_et) -> + message_et = new z.entity.MemberMessage() + message_et.user user_et + switch connection_et.status() + when z.user.ConnectionStatus.PENDING + message_et.member_message_type = z.message.SystemMessageType.CONNECTION_REQUEST + when z.user.ConnectionStatus.ACCEPTED + message_et.member_message_type = z.message.SystemMessageType.CONNECTION_ACCEPTED + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.NOTIFY, connection_et, message_et + + + ############################################################################### + # Users + ############################################################################### + + ### + Request account deletion. + @return [Promise] Promise that resolves when account deletion process has been initiated + ### + delete_me: => + @user_service.delete_self() + .then => + @logger.log @logger.levels.INFO, 'Account deletion initiated' + .catch (error) => + @logger.log @logger.levels.ERROR, "Unable to delete self: #{error}" + + ### + Get a user from the backend. + @param user_id [String] User ID + @param callback [Function] Function to be called on server return + ### + fetch_user_by_id: (user_id, callback) => + return callback?() if not user_id + + @fetch_users_by_id [user_id], (user_ets) -> + callback? user_ets?[0] + + ### + Get users from the backend. + @param user_ids [Array] User IDs + @param callback [Function] Function to be called on server return + ### + fetch_users_by_id: (user_ids, callback) => + user_ids = user_ids?.filter (user_id) -> + return user_id isnt undefined + + return callback? [] if not user_ids or user_ids.length is 0 + + # create chunks + fetched_user_ets = [] + chunks = z.util.array_chunks user_ids, z.config.MAXIMUM_USERS_PER_REQUEST + number_of_loaded_chunks = 0 + + for chunk in chunks + @user_service.get_users chunk, (response, error) => + number_of_loaded_chunks += 1 + if response + user_ets = @user_mapper.map_users_from_object response + fetched_user_ets = fetched_user_ets.concat user_ets + + if number_of_loaded_chunks is chunks.length + # If the difference is 1 then we most likely have a case with a suspended user + if user_ids.length isnt fetched_user_ets.length + fetched_user_ets = @_add_suspended_users user_ids, fetched_user_ets + @save_users fetched_user_ets + callback? fetched_user_ets + + ### + Find a local user. + @param user_id [String] User ID + @return [z.entity.User] Matching user entity + ### + find_user: (user_id) -> + return user_et for user_et in @users() when user_et.id is user_id + + ### + Get self user from backend. + @return [Promise] Promise that will resolve with the self user entity + ### + get_me: => + return new Promise (resolve, reject) => + @user_service.get_own_user() + .then (response) => + user_et = @user_mapper.map_user_from_object response + # TODO: This needs to be represented by a SelfUser class! + # Only the "self / own" user has a tracking ID & locale + user_et.tracking_id = response.tracking_id + user_et.locale = response.locale + @save_user user_et, true + resolve user_et + .catch (error) => + @logger.log @logger.levels.ERROR, "Unable to load self user: #{error}" + reject error + + ### + Check for user locally and fetch it from the server otherwise. + @param user_id [String] User ID + @param callback [Function] Function to be called on server return + ### + get_user_by_id: (user_id, callback) -> + if user_id is undefined + callback? null + + user_et = @find_user user_id + if not user_et + @fetch_user_by_id user_id, callback + else + callback? user_et + + return user_et + + ### + Check for users locally and fetch them from the server otherwise. + + @param user_ids [Array] User IDs + @param callback [Function] Function to be called on server return + @param offline [Boolean] Should we only look for cached contacts + ### + get_users_by_id: (user_ids = [], callback, offline = false) => + return callback? [] if user_ids.length is 0 + + known_user_ets = [] + unknown_user_ids = [] + + user_ids.forEach (user_id) => + user_et = @find_user user_id + + if user_et? + known_user_ets.push user_et + else + unknown_user_ids.push user_id + + if unknown_user_ids.length is 0 or offline + callback? known_user_ets + else + @fetch_users_by_id unknown_user_ids, (user_ets) -> + callback? known_user_ets.concat user_ets + + ### + Get user by name. + @param name [String] Name to search user for + @return [Array] Matching users + ### + get_user_by_name: (name) => + user_ets = [] + + for user_et in @users() + if user_et.connected() and (user_et.email() is name or z.util.compare_names user_et.name(), name) + user_ets.push user_et + + return user_ets.sort z.util.sort_user_by_name + + ### + Is the user the logged in user. + @param user_id [z.entity.User or String] User entity or user ID + @return [Boolean] Is the user the logged in user + ### + is_me: (user_id) -> + user_id = user_id.id if not _.isString user_id + return @self().id is user_id + + ### + Save a user. + @param user_et [z.entity.User] User entity to be stored + @param is_me [Boolean] Is the user entity the self user + ### + save_user: (user_et, is_me = false) -> + @users.push user_et if not @user_exists(user_et.id) + + if is_me + user_et.is_me = true + @self user_et + + return user_et + + ### + Save multiple users at once. + @param user_ets [Array] Array of user entities to be stored + ### + save_users: (user_ets) -> + new_user_ets = (user_et for user_et in user_ets when not @user_exists user_et.id) + z.util.ko_array_push_all @users, new_user_ets + return new_user_ets + + ### + Update a local user from the backend by ID. + @param user_id [String] User ID + ### + update_user_by_id: (user_id) => + @get_user_by_id user_id, (old_user_et) => + @user_service.get_user_by_id user_id, (new_user_et) => + @user_mapper.update_user_from_object old_user_et, new_user_et + + ### + Check if the user is already stored. + @param user_id [String] User ID + @return [Boolean] Is the user already stored + ### + user_exists: (user_id) -> + user_et = @find_user user_id + return !!user_et + + ### + Add user entities for suspended users. + + @param user_ids [Array] Requested user IDs + @param user_ets [Array] User entities returned by backend + @return [Array] User entities to be returned + ### + _add_suspended_users: (user_ids, user_ets) -> + for user_id in user_ids + if not (user_ets.find (element) -> return true if element.id is user_id) + user_et = new z.entity.User(user_id) + user_et.name z.localization.Localizer.get_text z.string.nonexistent_user + user_ets.push user_et + return user_ets + + ############################################################################### + # Profile + ############################################################################### + + ### + Change the accent color. + @param accent_id [Integer] New accent color + ### + change_accent_color: (accent_id) -> + @user_service.update_own_user_profile {accent_id: accent_id}, (response, error) => + @self().accent_id accent_id if not error? + + ### + Change username. + @param name [String] New user name + ### + change_username: (name) -> + if name.length >= z.config.MINIMUM_USERNAME_LENGTH + @user_service.update_own_user_profile {name: name}, (response, error) => + @self().name name if not error? + + ### + Change the profile image. + @param picture [String, Object] New user picture + @param on_success [Function] Function to be executed on success + ### + change_picture: (picture, on_success) -> + @asset_service.upload_profile_image @self().id, picture, (upload_response, error) => + if upload_response + @user_service.update_own_user_profile {picture: upload_response}, (update_response, error) => + if not error? + @user_update {user: {id: @self().id, picture: upload_response}} + on_success?() + else + @logger.log @logger.levels.ERROR, "Error during profile image upload: #{error.message}", error + + + ############################################################################### + # Properties + ############################################################################### + + ### + Initialize properties on app startup. + ### + init_properties: => + return new Promise (resolve, reject) => + @user_service.get_user_properties() + .then (response) => + if response.includes z.config.PROPERTIES_KEY + @user_service.get_user_properties_by_key z.config.PROPERTIES_KEY + .then (response) => + $.extend true, @properties, response + @logger.log @logger.levels.INFO, 'Loaded user properties', @properties + else + @logger.log @logger.levels.INFO, 'User has no saved properties, using defaults' + .then => + amplify.publish z.event.WebApp.PROPERTIES.UPDATED, @properties + amplify.publish z.event.WebApp.ANALYTICS.INIT, @properties, @self() + resolve @properties + .catch (error) => + error = new Error "Failed to initialize user properties: #{error}" + @logger.log @logger.levels.ERROR, error.message, error + reject @properties + + properties_updated: (properties) -> + if properties.enable_debugging + amplify.publish z.util.Logger::LOG_ON_DEBUG, properties.enable_debugging + + ### + Save the user properties. + ### + save_properties: (key, value) => + @user_service.change_user_properties_by_key z.config.PROPERTIES_KEY, @properties + .then => + @logger.log @logger.levels.INFO, "Saved updated settings: '#{key}' - '#{value}'" + .catch (error) => + @logger.log @logger.levels.ERROR, 'Saving updated settings failed', error + + ### + Save timestamp for Google Contacts import. + @param timestamp [String] Timestamp to be saved + ### + save_property_contact_import_google: (timestamp) => + @properties.contact_import.google = timestamp + @save_properties 'contact_import.google', timestamp + .then -> amplify.publish z.event.WebApp.PROPERTIES.UPDATE.GOOGLE, timestamp + + ### + Save timestamp for OSX Contacts import. + @param timestamp [String] Timestamp to be saved + ### + save_property_contact_import_osx: (timestamp) => + @properties.contact_import.osx = timestamp + @save_properties 'contact_import.osx', timestamp + .then -> amplify.publish z.event.WebApp.PROPERTIES.UPDATE.OSX_CONTACTS, timestamp + + ### + Save data settings. + @param is_enabled [String] Data setting to be saved + ### + save_property_data_settings: (is_enabled) => + return if @properties.settings.privacy.report_errors is is_enabled + @properties.settings.privacy.report_errors = is_enabled + @properties.settings.privacy.improve_wire = is_enabled + @save_properties 'settings.privacy', is_enabled + .then -> amplify.publish z.event.WebApp.PROPERTIES.UPDATE.SEND_DATA, is_enabled + + ### + Save debug logging setting. + @param is_enabled [Boolean] Should debug logging be enabled despite domain + ### + save_property_enable_debugging: (is_enabled) => + return if @properties.enable_debugging is is_enabled + @properties.enable_debugging = is_enabled + @save_properties 'enable_debugging', is_enabled + .then -> amplify.publish z.util.Logger::LOG_ON_DEBUG, is_enabled + + ### + Save timestamp for Google Contacts import. + ### + save_property_has_created_conversation: => + @properties.has_created_conversation = true + @save_properties 'has_created_conversation', @properties.has_created_conversation + .then -> amplify.publish z.event.WebApp.PROPERTIES.UPDATE.HAS_CREATED_CONVERSATION + + ### + Save audio settings. + @param sound_alerts [String] Audio setting to be saved + ### + save_property_sound_alerts: (sound_alerts) => + if @properties.settings.sound.alerts isnt sound_alerts + @properties.settings.sound.alerts = sound_alerts + @save_properties 'settings.sound.alerts', @properties.settings.sound.alerts + .then -> amplify.publish z.event.WebApp.PROPERTIES.UPDATE.SOUND_ALERTS, sound_alerts + + + ############################################################################### + # Tracking helpers + ############################################################################### + + ### + Count of connections. + @return [Integer] Number of connections + ### + get_number_of_connections: -> + amount = + incoming: 0 + outgoing: 0 + + for connection_et in @connections() + if connection_et.status() is z.user.ConnectionStatus.PENDING + amount.incoming += 1 + else if connection_et.status() is z.user.ConnectionStatus.SENT + amount.outgoing += 1 + + return amount diff --git a/app/script/user/UserService.coffee b/app/script/user/UserService.coffee new file mode 100644 index 00000000000..876e23e1fad --- /dev/null +++ b/app/script/user/UserService.coffee @@ -0,0 +1,315 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.user ?= {} + +### +User service for all user and connection calls to the backend REST API. +@todo move everything that is not users or self (e.g. login,register,activate) to AuthService? +### +class z.user.UserService + URL_CONNECTIONS: '/connections' + URL_PROPERTIES: '/properties' + URL_SELF: '/self' + URL_USERS: '/users' + ### + Construct a new User Service. + @param client [z.service.Client] Client for the API calls + ### + constructor: (@client) -> + @logger = new z.util.Logger 'z.user.UserService', z.config.LOGGER.OPTIONS + + + ############################################################################### + # Connections + ############################################################################### + + ### + Create a connection request to another user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/createConnection + + @param user_id [String] User ID of the user to request a connection with + @param name [String] Name of the conversation being initiated (1 - 256 characters) + @param message [String] The initial message in the request (1 - 256 characters) + @return [Promise] Promise that resolves when the connection request was created + ### + create_connection: (user_id, name, message) -> + payload = + user: user_id + name: name + message: message + + @client.send_json + type: 'POST' + url: @client.create_url @URL_CONNECTIONS + data: payload + + ### + Retrieves a list of connections to other users. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/connections + @note The list is already pre-ordered by the backend, so in order to fetch more connections + than the limit, you only have to pass the User ID (which is not from the self user) + of the last connection item from the received list. + + @param limit [Integer] Number of results to return (default 100, max 500) + @param user_id [String] User ID to start from + @return [Promise] Promise that resolves with user connections + ### + get_own_connections: (limit = 500, user_id = undefined) -> + @client.send_request + type: 'GET' + data: + size: limit + start: user_id + url: @client.create_url @URL_CONNECTIONS + + ### + Updates a connection to another user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/updateConnection + @example status: ['accepted', 'blocked', 'pending', 'ignored', 'sent' or 'cancelled'] + + @param user_id [String] User ID of the other user + @param status [z.user.ConnectionStatus] New relation status + @return [Promise] Promise that resolves when the status was updated + ### + update_connection_status: (user_id, status) -> + @client.send_json + type: 'PUT' + url: @client.create_url "/connections/#{user_id}" + data: + status: status + + + ############################################################################### + # Password reset + ############################################################################### + + ### + Initiate a password reset. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/beginPasswordReset + + @param email [String] Email address + @param phone [String] E.164 formatted phone number + @return [Promise] Promise that resolves when password reset process has been triggered + ### + initiate_password_reset: (email, phone_number) -> + @client.send_json + type: 'POST' + url: @client.create_url '/password-reset' + data: + email: email + phone: phone_number + + ############################################################################### + # Profile + ############################################################################### + + ### + Get your profile. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/self + @return [Promise] Promise that will resolve with the self user + ### + get_own_user: -> + @client.send_request + type: 'GET' + url: @client.create_url @URL_SELF + + ### + Update your profile. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/updateSelf + + @param data [Object] Updated user profile information + @option data [Integer] accent_id + @option data [String] name + @option data [Array] picture + @param callback [Function] Function to be called on server return + ### + update_own_user_profile: (data, callback) -> + @client.send_json + type: 'PUT' + url: @client.create_url @URL_SELF + data: data + callback: callback + + ### + Change user email. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/changeEmail + + @param email [String] New email address for the user + @return [Promise] Promise that resolves when email changing process has been started on backend + ### + change_own_email: (email) -> + @client.send_json + type: 'PUT' + url: @client.create_url '/self/email' + data: + email: email + + ### + Change user password. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/changePassword + + @param new_password [String] New user password + @param old_password [String] Old password of the user (optional) + @return [Promise] Promise that resolves when password has been changed on backend + ### + change_own_password: (new_password, old_password) -> + @client.send_json + type: 'PUT' + url: @client.create_url '/self/password' + data: + new_password: new_password + old_password: old_password + + ### + Change your phone number. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/changePhone + + @param phone_number [String] Phone number in E.164 format + @return [Promise] Promise that resolves when phone number change process has been started on backend + ### + change_own_phone_number: (phone_number) -> + @client.send_json + type: 'PUT' + url: @client.create_url '/self/phone' + data: + phone: phone_number + + + ############################################################################### + # Properties + ############################################################################### + + ### + List all property keys stored for the user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/listPropertyKeys + @return [Promise] Promise that resolves with a list of the property keys stored for the user + ### + get_user_properties: -> + @client.send_request + type: 'GET' + url: @client.create_url @URL_PROPERTIES + + ### + Clear all properties store for the user. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/clearProperties + @return [Promise] Promise that resolves when properties for user have been cleared + ### + clear_user_properties: -> + @client.send_request + type: 'DELETE' + url: @client.create_url @URL_PROPERTIES + + ### + Get a property value stored for a key. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/getProperty + + @param key [String] Key used to store user properties + @return [Promise] Promise that resolves with the properties for the given key + ### + get_user_properties_by_key: (key) -> + @client.send_request + type: 'GET' + url: @client.create_url "/properties/#{key}" + + + ### + Set a property value for a key. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/setProperty + + @param key [String] Key used to store user properties + @param properties [Object] Payload to be stored + ### + change_user_properties_by_key: (key, properties) -> + @client.send_json + type: 'PUT' + url: @client.create_url "/properties/#{key}" + data: properties + + ### + Delete a property. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/deleteProperty + + @param key [String] Key used to store user properties + @param callback [Function] Function to be called on server return + ### + delete_user_properties_by_key: (key) -> + @client.send_request + type: 'DELETE' + url: @client.create_url "/properties/#{key}" + callback: callback + + + ############################################################################### + # Users + ############################################################################### + + ### + Delete self user. + @return [Promise] Promise that resolves when account deletion has been initiated + ### + delete_self: -> + @client.send_json + type: 'DELETE' + url: @client.create_url @URL_SELF + data: + todo: 'Change this to normal request!' + + ### + Check if a user ID exists. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/userExists + @example "496d0d21-0b05-49b5-8087-de94f3465b7b" + + @param user_id [String] User ID + @param callback [Function] Function to be called on server return + ### + is_existing_user: (user_id, callback) -> + @client.send_request + type: 'HEAD' + url: @client.create_url "/users/#{user_id}" + callback: callback + + ### + Get a set of users. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/users + @example ['0bb84213-8cc2-4bb1-9e0b-b8dd522396d5', '15ede065-72b3-433a-9917-252f076ed031'] + + @param users [Array] ID of users to be fetched + @param callback [Function] Function to be called on server return + ### + get_users: (users, callback) -> + @client.send_request + type: 'GET' + url: @client.create_url @URL_USERS + data: + ids: users.join ',' + callback: callback + + ### + Get a user by ID. + @see https://staging-nginz-https.zinfra.io/swagger-ui/#!/users/user + + @param user_id [String] User ID + @param callback [Function] Function to be called on server return + ### + get_user_by_id: (user_id, callback) -> + @client.send_request + type: 'GET' + url: @client.create_url "/users/#{user_id}" + callback: callback diff --git a/app/script/util/ArrayUtil.coffee b/app/script/util/ArrayUtil.coffee new file mode 100644 index 00000000000..8d167cfc689 --- /dev/null +++ b/app/script/util/ArrayUtil.coffee @@ -0,0 +1,72 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} +z.util.ArrayUtil ?= {} + +### +Moves an element from one place to another by it's index. +This change happens in place which means that the array is modified immediately. + +todo: nice idea, DSL with "move_element.from(...).to(...).in(...)" + +@param from [Number] Source index of the element which should be moved +@param to [Number] Destination index of the element which should be moved +@param array [Array] Array where to move the array +@return [Array] Given array +### +z.util.ArrayUtil.move_element = (from, to, array) -> + element = array[from] + array.splice from, 1 + array.splice to, 0, element + +### +Returns random element + +@param array [Array] source +@return [Object] random element +### +z.util.ArrayUtil.random_element = (array) -> + array[Math.floor(Math.random() * array.length)] + +z.util.ArrayUtil.contains = (array, value) -> + return array.indexOf(value) > -1 + +### +Interpolates an array of numbers using linear interpolation + +@param array [Array] source +@param length [Number] new length +@return [Array] new array with interpolated values +### +z.util.ArrayUtil.interpolate = (array, length) -> + new_array = [] + scale_factor = (array.length - 1) / (length - 1) + + new_array[0] = array[0] + new_array[length - 1] = array[array.length - 1] + + for i in [1...length - 1] + original_index = i * scale_factor + before = Math.floor(original_index).toFixed() + after = Math.ceil(original_index).toFixed() + point = original_index - before + new_array[i] = array[before] + (array[after] - array[before]) * point # linear interpolation + + return new_array diff --git a/app/script/util/BrowserPermissionType.coffee b/app/script/util/BrowserPermissionType.coffee new file mode 100644 index 00000000000..b92201a8780 --- /dev/null +++ b/app/script/util/BrowserPermissionType.coffee @@ -0,0 +1,27 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + + +# https://developer.mozilla.org/en-US/docs/Web/API/Notification.permission#Return_Value +z.util.BrowserPermissionType = + DEFAULT: 'default' + DENIED: 'denied' + GRANTED: 'granted' diff --git a/app/script/util/Comparison.coffee b/app/script/util/Comparison.coffee new file mode 100644 index 00000000000..883272c229e --- /dev/null +++ b/app/script/util/Comparison.coffee @@ -0,0 +1,31 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +z.util.Comparison = do -> + + public_methods = + max_number: (a, b) -> + return a if window.isNaN(b) and not window.isNaN(a) + return b if window.isNaN(a) and not window.isNaN(b) + return if a >= b then a else b + min_number: (a, b) -> + return a if window.isNaN(b) and not window.isNaN(a) + return b if window.isNaN(a) and not window.isNaN(b) + return if a >= b then b else a + + return public_methods diff --git a/app/script/util/CountryCodes.coffee b/app/script/util/CountryCodes.coffee new file mode 100644 index 00000000000..c53eca87ef7 --- /dev/null +++ b/app/script/util/CountryCodes.coffee @@ -0,0 +1,1511 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +z.util.CountryCodes = do -> + + ### + Get the country code matching an ISO name + + @param iso_name [String] ISO standard country name + + @return [Integer] Matching country code + ### + get_country_code = (iso_name) -> + for country in COUNTRY_CODES + return country.code if country.iso is iso_name + + ### + Get the full-text country name matching an ISO name + + @param iso_name [String] ISO standard country name + + @return [String] Matching full-text country name + ### + get_country_name = (iso_name) -> + for country in COUNTRY_CODES + return country.name if country.iso is iso_name + + ### + Get the country code matching to an ISO name + + @param code [Integer] Country code + + @return [String] Returns the ISO standard country name of the most populated country with the matching country code + ### + get_country_by_code = (code) -> + countries = [] + + for country in COUNTRY_CODES + countries.push country if country.code is window.parseInt code, 10 + + countries = countries.sort (country_a, country_b) -> + return country_a.population - country_b.population + + return countries.pop()?.iso + + COUNTRY_CODES = [ + { + 'code': 93 + 'iso': 'AF' + 'name': 'Afghanistan' + 'population': 31 + } + { + 'code': 355 + 'iso': 'AL' + 'name': 'Albania' + 'population': 2 + } + { + 'code': 213 + 'iso': 'DZ' + 'name': 'Algeria' + 'population': 34 + } + { + 'code': 1684 + 'iso': 'AS' + 'name': 'American Samoa' + 'population': 0 + } + { + 'code': 376 + 'iso': 'AD' + 'name': 'Andorra' + 'population': 0 + } + { + 'code': 244 + 'iso': 'AO' + 'name': 'Angola' + 'population': 13 + } + { + 'code': 1264 + 'iso': 'AI' + 'name': 'Anguilla' + 'population': 0 + } + { + 'code': 672 + 'iso': 'AQ' + 'name': 'Antarctica' + 'population': 0 + } + { + 'code': 1268 + 'iso': 'AG' + 'name': 'Antigua and Barbuda' + 'population': 0 + } + { + 'code': 54 + 'iso': 'AR' + 'name': 'Argentina' + 'population': 41 + } + { + 'code': 374 + 'iso': 'AM' + 'name': 'Armenia' + 'population': 2 + } + { + 'code': 297 + 'iso': 'AW' + 'name': 'Aruba' + 'population': 0 + } + { + 'code': 61 + 'iso': 'AU' + 'name': 'Australia' + 'population': 21 + } + { + 'code': 43 + 'iso': 'AT' + 'name': 'Austria' + 'population': 8 + } + { + 'code': 994 + 'iso': 'AZ' + 'name': 'Azerbaijan' + 'population': 8 + } + { + 'code': 1242 + 'iso': 'BS' + 'name': 'Bahamas' + 'population': 0 + } + { + 'code': 973 + 'iso': 'BH' + 'name': 'Bahrain' + 'population': 0 + } + { + 'code': 880 + 'iso': 'BD' + 'name': 'Bangladesh' + 'population': 171 + } + { + 'code': 1246 + 'iso': 'BB' + 'name': 'Barbados' + 'population': 0 + } + { + 'code': 375 + 'iso': 'BY' + 'name': 'Belarus' + 'population': 9 + } + { + 'code': 32 + 'iso': 'BE' + 'name': 'Belgium' + 'population': 10 + } + { + 'code': 501 + 'iso': 'BZ' + 'name': 'Belize' + 'population': 0 + } + { + 'code': 229 + 'iso': 'BJ' + 'name': 'Benin' + 'population': 9 + } + { + 'code': 1441 + 'iso': 'BM' + 'name': 'Bermuda' + 'population': 0 + } + { + 'code': 975 + 'iso': 'BT' + 'name': 'Bhutan' + 'population': 0 + } + { + 'code': 591 + 'iso': 'BO' + 'name': 'Bolivia' + 'population': 9 + } + { + 'code': 387 + 'iso': 'BA' + 'name': 'Bosnia and Herzegovina' + 'population': 4 + } + { + 'code': 267 + 'iso': 'BW' + 'name': 'Botswana' + 'population': 2 + } + { + 'code': 55 + 'iso': 'BR' + 'name': 'Brazil' + 'population': 201 + } + { + 'code': 246 + 'iso': 'IO' + 'name': 'British Indian Ocean Territory' + 'population': 0 + } + { + 'code': 1284 + 'iso': 'VG' + 'name': 'British Virgin Islands' + 'population': 0 + } + { + 'code': 673 + 'iso': 'BN' + 'name': 'Brunei' + 'population': 0 + } + { + 'code': 359 + 'iso': 'BG' + 'name': 'Bulgaria' + 'population': 7 + } + { + 'code': 226 + 'iso': 'BF' + 'name': 'Burkina Faso' + 'population': 16 + } + { + 'code': 257 + 'iso': 'BI' + 'name': 'Burundi' + 'population': 9 + } + { + 'code': 855 + 'iso': 'KH' + 'name': 'Cambodia' + 'population': 14 + } + { + 'code': 237 + 'iso': 'CM' + 'name': 'Cameroon' + 'population': 19 + } + { + 'code': 1 + 'iso': 'CA' + 'name': 'Canada' + 'population': 33 + } + { + 'code': 238 + 'iso': 'CV' + 'name': 'Cape Verde' + 'population': 0 + } + { + 'code': 1345 + 'iso': 'KY' + 'name': 'Cayman Islands' + 'population': 0 + } + { + 'code': 236 + 'iso': 'CF' + 'name': 'Central African Republic' + 'population': 4 + } + { + 'code': 235 + 'iso': 'TD' + 'name': 'Chad' + 'population': 10 + } + { + 'code': 56 + 'iso': 'CL' + 'name': 'Chile' + 'population': 16 + } + { + 'code': 86 + 'iso': 'CN' + 'name': 'China' + 'population': 1376 + } + { + 'code': 61 + 'iso': 'CX' + 'name': 'Christmas Island' + 'population': 0 + } + { + 'code': 61 + 'iso': 'CC' + 'name': 'Cocos Islands' + 'population': 0 + } + { + 'code': 57 + 'iso': 'CO' + 'name': 'Colombia' + 'population': 47 + } + { + 'code': 269 + 'iso': 'KM' + 'name': 'Comoros' + 'population': 0 + } + { + 'code': 682 + 'iso': 'CK' + 'name': 'Cook Islands' + 'population': 0 + } + { + 'code': 506 + 'iso': 'CR' + 'name': 'Costa Rica' + 'population': 4 + } + { + 'code': 385 + 'iso': 'HR' + 'name': 'Croatia' + 'population': 4 + } + { + 'code': 53 + 'iso': 'CU' + 'name': 'Cuba' + 'population': 11 + } + { + 'code': 599 + 'iso': 'CW' + 'name': 'Curacao' + 'population': 0 + } + { + 'code': 357 + 'iso': 'CY' + 'name': 'Cyprus' + 'population': 1 + } + { + 'code': 420 + 'iso': 'CZ' + 'name': 'Czech Republic' + 'population': 10 + } + { + 'code': 243 + 'iso': 'CD' + 'name': 'Democratic Republic of the Congo' + 'population': 81 + } + { + 'code': 45 + 'iso': 'DK' + 'name': 'Denmark' + 'population': 5 + } + { + 'code': 253 + 'iso': 'DJ' + 'name': 'Djibouti' + 'population': 0 + } + { + 'code': 1767 + 'iso': 'DM' + 'name': 'Dominica' + 'population': 0 + } + { + 'code': 1809 + 'iso': 'DO' + 'name': 'Dominican Republic' + 'population': 9 + } + { + 'code': 670 + 'iso': 'TL' + 'name': 'East Timor' + 'population': 1 + } + { + 'code': 593 + 'iso': 'EC' + 'name': 'Ecuador' + 'population': 14 + } + { + 'code': 20 + 'iso': 'EG' + 'name': 'Egypt' + 'population': 91 + } + { + 'code': 503 + 'iso': 'SV' + 'name': 'El Salvador' + 'population': 6 + } + { + 'code': 240 + 'iso': 'GQ' + 'name': 'Equatorial Guinea' + 'population': 1 + } + { + 'code': 291 + 'iso': 'ER' + 'name': 'Eritrea' + 'population': 5 + } + { + 'code': 372 + 'iso': 'EE' + 'name': 'Estonia' + 'population': 1 + } + { + 'code': 251 + 'iso': 'ET' + 'name': 'Ethiopia' + 'population': 88 + } + { + 'code': 500 + 'iso': 'FK' + 'name': 'Falkland Islands' + 'population': 0 + } + { + 'code': 298 + 'iso': 'FO' + 'name': 'Faroe Islands' + 'population': 0 + } + { + 'code': 679 + 'iso': 'FJ' + 'name': 'Fiji' + 'population': 0 + } + { + 'code': 358 + 'iso': 'FI' + 'name': 'Finland' + 'population': 5 + } + { + 'code': 33 + 'iso': 'FR' + 'name': 'France' + 'population': 64 + } + { + 'code': 689 + 'iso': 'PF' + 'name': 'French Polynesia' + 'population': 0 + } + { + 'code': 241 + 'iso': 'GA' + 'name': 'Gabon' + 'population': 1 + } + { + 'code': 220 + 'iso': 'GM' + 'name': 'Gambia' + 'population': 1 + } + { + 'code': 995 + 'iso': 'GE' + 'name': 'Georgia' + 'population': 4 + } + { + 'code': 49 + 'iso': 'DE' + 'name': 'Germany' + 'population': 81 + } + { + 'code': 233 + 'iso': 'GH' + 'name': 'Ghana' + 'population': 24 + } + { + 'code': 350 + 'iso': 'GI' + 'name': 'Gibraltar' + 'population': 0 + } + { + 'code': 30 + 'iso': 'GR' + 'name': 'Greece' + 'population': 11 + } + { + 'code': 299 + 'iso': 'GL' + 'name': 'Greenland' + 'population': 0 + } + { + 'code': 1473 + 'iso': 'GD' + 'name': 'Grenada' + 'population': 0 + } + { + 'code': 1671 + 'iso': 'GU' + 'name': 'Guam' + 'population': 0 + } + { + 'code': 502 + 'iso': 'GT' + 'name': 'Guatemala' + 'population': 13 + } + { + 'code': 441481 + 'iso': 'GG' + 'name': 'Guernsey' + 'population': 0 + } + { + 'code': 224 + 'iso': 'GN' + 'name': 'Guinea' + 'population': 10 + } + { + 'code': 245 + 'iso': 'GW' + 'name': 'Guinea-Bissau' + 'population': 1 + } + { + 'code': 592 + 'iso': 'GY' + 'name': 'Guyana' + 'population': 0 + } + { + 'code': 509 + 'iso': 'HT' + 'name': 'Haiti' + 'population': 9 + } + { + 'code': 504 + 'iso': 'HN' + 'name': 'Honduras' + 'population': 7 + } + { + 'code': 852 + 'iso': 'HK' + 'name': 'Hong Kong' + 'population': 6 + } + { + 'code': 36 + 'iso': 'HU' + 'name': 'Hungary' + 'population': 9 + } + { + 'code': 354 + 'iso': 'IS' + 'name': 'Iceland' + 'population': 0 + } + { + 'code': 91 + 'iso': 'IN' + 'name': 'India' + 'population': 1173 + } + { + 'code': 62 + 'iso': 'ID' + 'name': 'Indonesia' + 'population': 242 + } + { + 'code': 98 + 'iso': 'IR' + 'name': 'Iran' + 'population': 76 + } + { + 'code': 964 + 'iso': 'IQ' + 'name': 'Iraq' + 'population': 29 + } + { + 'code': 353 + 'iso': 'IE' + 'name': 'Ireland' + 'population': 4 + } + { + 'code': 441624 + 'iso': 'IM' + 'name': 'Isle of Man' + 'population': 0 + } + { + 'code': 972 + 'iso': 'IL' + 'name': 'Israel' + 'population': 7 + } + { + 'code': 39 + 'iso': 'IT' + 'name': 'Italy' + 'population': 60 + } + { + 'code': 225 + 'iso': 'CI' + 'name': 'Ivory Coast' + 'population': 21 + } + { + 'code': 1876 + 'iso': 'JM' + 'name': 'Jamaica' + 'population': 2 + } + { + 'code': 81 + 'iso': 'JP' + 'name': 'Japan' + 'population': 127 + } + { + 'code': 441534 + 'iso': 'JE' + 'name': 'Jersey' + 'population': 0 + } + { + 'code': 962 + 'iso': 'JO' + 'name': 'Jordan' + 'population': 6 + } + { + 'code': 7 + 'iso': 'KZ' + 'name': 'Kazakhstan' + 'population': 15 + } + { + 'code': 254 + 'iso': 'KE' + 'name': 'Kenya' + 'population': 40 + } + { + 'code': 686 + 'iso': 'KI' + 'name': 'Kiribati' + 'population': 0 + } + { + 'code': 383 + 'iso': 'XK' + 'name': 'Kosovo' + 'population': 1 + } + { + 'code': 965 + 'iso': 'KW' + 'name': 'Kuwait' + 'population': 2 + } + { + 'code': 996 + 'iso': 'KG' + 'name': 'Kyrgyzstan' + 'population': 5 + } + { + 'code': 856 + 'iso': 'LA' + 'name': 'Laos' + 'population': 6 + } + { + 'code': 371 + 'iso': 'LV' + 'name': 'Latvia' + 'population': 2 + } + { + 'code': 961 + 'iso': 'LB' + 'name': 'Lebanon' + 'population': 4 + } + { + 'code': 266 + 'iso': 'LS' + 'name': 'Lesotho' + 'population': 1 + } + { + 'code': 231 + 'iso': 'LR' + 'name': 'Liberia' + 'population': 3 + } + { + 'code': 218 + 'iso': 'LY' + 'name': 'Libya' + 'population': 6 + } + { + 'code': 423 + 'iso': 'LI' + 'name': 'Liechtenstein' + 'population': 0 + } + { + 'code': 370 + 'iso': 'LT' + 'name': 'Lithuania' + 'population': 2 + } + { + 'code': 352 + 'iso': 'LU' + 'name': 'Luxembourg' + 'population': 0 + } + { + 'code': 853 + 'iso': 'MO' + 'name': 'Macao' + 'population': 0 + } + { + 'code': 389 + 'iso': 'MK' + 'name': 'Macedonia' + 'population': 2 + } + { + 'code': 261 + 'iso': 'MG' + 'name': 'Madagascar' + 'population': 21 + } + { + 'code': 265 + 'iso': 'MW' + 'name': 'Malawi' + 'population': 15 + } + { + 'code': 60 + 'iso': 'MY' + 'name': 'Malaysia' + 'population': 28 + } + { + 'code': 960 + 'iso': 'MV' + 'name': 'Maldives' + 'population': 0 + } + { + 'code': 223 + 'iso': 'ML' + 'name': 'Mali' + 'population': 13 + } + { + 'code': 356 + 'iso': 'MT' + 'name': 'Malta' + 'population': 0 + } + { + 'code': 692 + 'iso': 'MH' + 'name': 'Marshall Islands' + 'population': 0 + } + { + 'code': 222 + 'iso': 'MR' + 'name': 'Mauritania' + 'population': 3 + } + { + 'code': 230 + 'iso': 'MU' + 'name': 'Mauritius' + 'population': 1 + } + { + 'code': 262 + 'iso': 'YT' + 'name': 'Mayotte' + 'population': 0 + } + { + 'code': 52 + 'iso': 'MX' + 'name': 'Mexico' + 'population': 112 + } + { + 'code': 691 + 'iso': 'FM' + 'name': 'Micronesia' + 'population': 0 + } + { + 'code': 373 + 'iso': 'MD' + 'name': 'Moldova' + 'population': 4 + } + { + 'code': 377 + 'iso': 'MC' + 'name': 'Monaco' + 'population': 0 + } + { + 'code': 976 + 'iso': 'MN' + 'name': 'Mongolia' + 'population': 3 + } + { + 'code': 382 + 'iso': 'ME' + 'name': 'Montenegro' + 'population': 0 + } + { + 'code': 1664 + 'iso': 'MS' + 'name': 'Montserrat' + 'population': 0 + } + { + 'code': 212 + 'iso': 'MA' + 'name': 'Morocco' + 'population': 33 + } + { + 'code': 258 + 'iso': 'MZ' + 'name': 'Mozambique' + 'population': 22 + } + { + 'code': 95 + 'iso': 'MM' + 'name': 'Myanmar' + 'population': 53 + } + { + 'code': 264 + 'iso': 'NA' + 'name': 'Namibia' + 'population': 2 + } + { + 'code': 674 + 'iso': 'NR' + 'name': 'Nauru' + 'population': 0 + } + { + 'code': 977 + 'iso': 'NP' + 'name': 'Nepal' + 'population': 28 + } + { + 'code': 31 + 'iso': 'NL' + 'name': 'Netherlands' + 'population': 16 + } + { + 'code': 599 + 'iso': 'AN' + 'name': 'Netherlands Antilles' + 'population': 0 + } + { + 'code': 687 + 'iso': 'NC' + 'name': 'New Caledonia' + 'population': 0 + } + { + 'code': 64 + 'iso': 'NZ' + 'name': 'New Zealand' + 'population': 4 + } + { + 'code': 505 + 'iso': 'NI' + 'name': 'Nicaragua' + 'population': 5 + } + { + 'code': 227 + 'iso': 'NE' + 'name': 'Niger' + 'population': 15 + } + { + 'code': 234 + 'iso': 'NG' + 'name': 'Nigeria' + 'population': 182 + } + { + 'code': 683 + 'iso': 'NU' + 'name': 'Niue' + 'population': 0 + } + { + 'code': 850 + 'iso': 'KP' + 'name': 'North Korea' + 'population': 22 + } + { + 'code': 1670 + 'iso': 'MP' + 'name': 'Northern Mariana Islands' + 'population': 0 + } + { + 'code': 47 + 'iso': 'NO' + 'name': 'Norway' + 'population': 5 + } + { + 'code': 968 + 'iso': 'OM' + 'name': 'Oman' + 'population': 2 + } + { + 'code': 92 + 'iso': 'PK' + 'name': 'Pakistan' + 'population': 184 + } + { + 'code': 680 + 'iso': 'PW' + 'name': 'Palau' + 'population': 0 + } + { + 'code': 970 + 'iso': 'PS' + 'name': 'Palestine' + 'population': 3 + } + { + 'code': 507 + 'iso': 'PA' + 'name': 'Panama' + 'population': 3 + } + { + 'code': 675 + 'iso': 'PG' + 'name': 'Papua New Guinea' + 'population': 6 + } + { + 'code': 595 + 'iso': 'PY' + 'name': 'Paraguay' + 'population': 6 + } + { + 'code': 51 + 'iso': 'PE' + 'name': 'Peru' + 'population': 29 + } + { + 'code': 63 + 'iso': 'PH' + 'name': 'Philippines' + 'population': 102 + } + { + 'code': 64 + 'iso': 'PN' + 'name': 'Pitcairn' + 'population': 0 + } + { + 'code': 48 + 'iso': 'PL' + 'name': 'Poland' + 'population': 38 + } + { + 'code': 351 + 'iso': 'PT' + 'name': 'Portugal' + 'population': 10 + } + { + 'code': 1787 + 'iso': 'PR' + 'name': 'Puerto Rico' + 'population': 3 + } + { + 'code': 974 + 'iso': 'QA' + 'name': 'Qatar' + 'population': 0 + } + { + 'code': 242 + 'iso': 'CG' + 'name': 'Republic of the Congo' + 'population': 3 + } + { + 'code': 262 + 'iso': 'RE' + 'name': 'Reunion' + 'population': 0 + } + { + 'code': 40 + 'iso': 'RO' + 'name': 'Romania' + 'population': 21 + } + { + 'code': 7 + 'iso': 'RU' + 'name': 'Russia' + 'population': 140 + } + { + 'code': 250 + 'iso': 'RW' + 'name': 'Rwanda' + 'population': 11 + } + { + 'code': 590 + 'iso': 'BL' + 'name': 'Saint Barthelemy' + 'population': 0 + } + { + 'code': 290 + 'iso': 'SH' + 'name': 'Saint Helena' + 'population': 0 + } + { + 'code': 1869 + 'iso': 'KN' + 'name': 'Saint Kitts and Nevis' + 'population': 0 + } + { + 'code': 1758 + 'iso': 'LC' + 'name': 'Saint Lucia' + 'population': 0 + } + { + 'code': 590 + 'iso': 'MF' + 'name': 'Saint Martin' + 'population': 0 + } + { + 'code': 508 + 'iso': 'PM' + 'name': 'Saint Pierre and Miquelon' + 'population': 0 + } + { + 'code': 1784 + 'iso': 'VC' + 'name': 'Saint Vincent and the Grenadines' + 'population': 0 + } + { + 'code': 685 + 'iso': 'WS' + 'name': 'Samoa' + 'population': 0 + } + { + 'code': 378 + 'iso': 'SM' + 'name': 'San Marino' + 'population': 0 + } + { + 'code': 239 + 'iso': 'ST' + 'name': 'Sao Tome and Principe' + 'population': 0 + } + { + 'code': 966 + 'iso': 'SA' + 'name': 'Saudi Arabia' + 'population': 25 + } + { + 'code': 221 + 'iso': 'SN' + 'name': 'Senegal' + 'population': 12 + } + { + 'code': 381 + 'iso': 'RS' + 'name': 'Serbia' + 'population': 7 + } + { + 'code': 248 + 'iso': 'SC' + 'name': 'Seychelles' + 'population': 0 + } + { + 'code': 232 + 'iso': 'SL' + 'name': 'Sierra Leone' + 'population': 5 + } + { + 'code': 65 + 'iso': 'SG' + 'name': 'Singapore' + 'population': 4 + } + { + 'code': 1721 + 'iso': 'SX' + 'name': 'Sint Maarten' + 'population': 0 + } + { + 'code': 421 + 'iso': 'SK' + 'name': 'Slovakia' + 'population': 5 + } + { + 'code': 386 + 'iso': 'SI' + 'name': 'Slovenia' + 'population': 2 + } + { + 'code': 677 + 'iso': 'SB' + 'name': 'Solomon Islands' + 'population': 0 + } + { + 'code': 252 + 'iso': 'SO' + 'name': 'Somalia' + 'population': 10 + } + { + 'code': 27 + 'iso': 'ZA' + 'name': 'South Africa' + 'population': 54 + } + { + 'code': 82 + 'iso': 'KR' + 'name': 'South Korea' + 'population': 48 + } + { + 'code': 211 + 'iso': 'SS' + 'name': 'South Sudan' + 'population': 8 + } + { + 'code': 34 + 'iso': 'ES' + 'name': 'Spain' + 'population': 46 + } + { + 'code': 94 + 'iso': 'LK' + 'name': 'Sri Lanka' + 'population': 21 + } + { + 'code': 249 + 'iso': 'SD' + 'name': 'Sudan' + 'population': 35 + } + { + 'code': 597 + 'iso': 'SR' + 'name': 'Suriname' + 'population': 0 + } + { + 'code': 47 + 'iso': 'SJ' + 'name': 'Svalbard and Jan Mayen' + 'population': 0 + } + { + 'code': 268 + 'iso': 'SZ' + 'name': 'Swaziland' + 'population': 1 + } + { + 'code': 46 + 'iso': 'SE' + 'name': 'Sweden' + 'population': 9 + } + { + 'code': 41 + 'iso': 'CH' + 'name': 'Switzerland' + 'population': 8 + } + { + 'code': 963 + 'iso': 'SY' + 'name': 'Syria' + 'population': 22 + } + { + 'code': 886 + 'iso': 'TW' + 'name': 'Taiwan' + 'population': 22 + } + { + 'code': 992 + 'iso': 'TJ' + 'name': 'Tajikistan' + 'population': 7 + } + { + 'code': 255 + 'iso': 'TZ' + 'name': 'Tanzania' + 'population': 41 + } + { + 'code': 66 + 'iso': 'TH' + 'name': 'Thailand' + 'population': 67 + } + { + 'code': 228 + 'iso': 'TG' + 'name': 'Togo' + 'population': 6 + } + { + 'code': 690 + 'iso': 'TK' + 'name': 'Tokelau' + 'population': 0 + } + { + 'code': 676 + 'iso': 'TO' + 'name': 'Tonga' + 'population': 0 + } + { + 'code': 1868 + 'iso': 'TT' + 'name': 'Trinidad and Tobago' + 'population': 1 + } + { + 'code': 216 + 'iso': 'TN' + 'name': 'Tunisia' + 'population': 10 + } + { + 'code': 90 + 'iso': 'TR' + 'name': 'Turkey' + 'population': 77 + } + { + 'code': 993 + 'iso': 'TM' + 'name': 'Turkmenistan' + 'population': 4 + } + { + 'code': 1649 + 'iso': 'TC' + 'name': 'Turks and Caicos Islands' + 'population': 0 + } + { + 'code': 688 + 'iso': 'TV' + 'name': 'Tuvalu' + 'population': 0 + } + { + 'code': 1340 + 'iso': 'VI' + 'name': 'U.S. Virgin Islands' + 'population': 0 + } + { + 'code': 256 + 'iso': 'UG' + 'name': 'Uganda' + 'population': 33 + } + { + 'code': 380 + 'iso': 'UA' + 'name': 'Ukraine' + 'population': 45 + } + { + 'code': 971 + 'iso': 'AE' + 'name': 'United Arab Emirates' + 'population': 4 + } + { + 'code': 44 + 'iso': 'GB' + 'name': 'United Kingdom' + 'population': 65 + } + { + 'code': 1 + 'iso': 'US' + 'name': 'United States' + 'population': 324 + } + { + 'code': 598 + 'iso': 'UY' + 'name': 'Uruguay' + 'population': 3 + } + { + 'code': 998 + 'iso': 'UZ' + 'name': 'Uzbekistan' + 'population': 27 + } + { + 'code': 678 + 'iso': 'VU' + 'name': 'Vanuatu' + 'population': 0 + } + { + 'code': 379 + 'iso': 'VA' + 'name': 'Vatican' + 'population': 0 + } + { + 'code': 58 + 'iso': 'VE' + 'name': 'Venezuela' + 'population': 26 + } + { + 'code': 84 + 'iso': 'VN' + 'name': 'Vietnam' + 'population': 91 + } + { + 'code': 681 + 'iso': 'WF' + 'name': 'Wallis and Futuna' + 'population': 0 + } + { + 'code': 212 + 'iso': 'EH' + 'name': 'Western Sahara' + 'population': 0 + } + { + 'code': 967 + 'iso': 'YE' + 'name': 'Yemen' + 'population': 24 + } + { + 'code': 260 + 'iso': 'ZM' + 'name': 'Zambia' + 'population': 16 + } + { + 'code': 263 + 'iso': 'ZW' + 'name': 'Zimbabwe' + 'population': 12 + } + ] + + return {} = + get_country_code: get_country_code + get_country_name: get_country_name + get_country_by_code: get_country_by_code + COUNTRY_CODES: COUNTRY_CODES diff --git a/app/script/util/Crypto.coffee b/app/script/util/Crypto.coffee new file mode 100644 index 00000000000..72393180346 --- /dev/null +++ b/app/script/util/Crypto.coffee @@ -0,0 +1,42 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +z.util.Crypto = do -> + Hashing = {} + + # Jenkins's one-at-a-time hash + Hashing.joaat_hash = (str) -> + uint32 = window.uint32 + hash = uint32.toUint32 0 + key = str.toLowerCase() + + for i in [0..key.length - 1] + hash = uint32.addMod32 hash, uint32.toUint32 key.charCodeAt i + hash = uint32.addMod32 hash, uint32.shiftLeft hash, 10 + hash = uint32.xor hash, uint32.shiftRight hash, 6 + + hash = uint32.addMod32 hash, uint32.shiftLeft hash, 3 + hash = uint32.xor hash, uint32.shiftRight hash, 11 + hash = uint32.addMod32 hash, uint32.shiftLeft hash, 15 + + return hash + + public_methods = + Hashing: Hashing + + return public_methods diff --git a/app/script/util/DebugUtil.coffee b/app/script/util/DebugUtil.coffee new file mode 100644 index 00000000000..15ecce1fa48 --- /dev/null +++ b/app/script/util/DebugUtil.coffee @@ -0,0 +1,50 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +class z.util.DebugUtil + constructor: (@user_repository, @conversation_repository) -> + @logger = new z.util.Logger 'z.util.DebugUtil', z.config.LOGGER.OPTIONS + + _get_conversation_by_id: (conversation_id) -> + return new Promise (resolve) => + @conversation_repository.get_conversation_by_id conversation_id, (conversation_et) -> + resolve conversation_et + + _get_user_by_id: (user_id) -> + return new Promise (resolve) => + @user_repository.get_user_by_id user_id, (user_et) -> resolve user_et + + get_event_info: (event) -> + debug_information = + event: event + + return new Promise (resolve) => + @_get_conversation_by_id event.conversation + .then (conversation_et) => + debug_information.conversation = conversation_et + return @_get_user_by_id event.from + .then (user_et) => + debug_information.user = user_et + log_message = "Hey #{@user_repository.self().name()}, this is for you:" + @logger.log @logger.levels.WARN, log_message, debug_information + @logger.log @logger.levels.WARN, "Conversation: #{debug_information.conversation.name()}", debug_information.conversation + @logger.log @logger.levels.WARN, "From: #{debug_information.user.name()}", debug_information.user + resolve debug_information diff --git a/app/script/util/Environment.coffee b/app/script/util/Environment.coffee new file mode 100644 index 00000000000..3db4ead32a1 --- /dev/null +++ b/app/script/util/Environment.coffee @@ -0,0 +1,133 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +APP_ENV = + LOCALHOST: 'localhost' + PRODUCTION: 'app.wire.com' + PROD_NEXT: 'wire-webapp-prod-next.wire.com' + TACO: 'taco.wire.com' + VIRTUAL_HOST: 'wire.ms' # The domain "wire.ms" is our virtual host for testing contact uploads + +BROWSER_NAME = + CHROME: 'Chrome' + EDGE: 'Microsoft Edge' + ELECTRON: 'Electron' + FIREFOX: 'Firefox' + OPERA: 'Opera' + +PLATFORM_NAME = + MACINTOSH: 'Mac' + WINDOWS: 'Win' + +# TODO: We need to remove the "do ->" pattern and invoke the Environment ourselves! +# Otherwise we cannot override the "navigator.userAgent" in our integration tests to check the behaviour of this class. +z.util.Environment = do -> + _check = + is_chrome: -> + return platform.name is BROWSER_NAME.CHROME + is_edge: -> + return platform.name is BROWSER_NAME.EDGE + is_firefox: -> + return platform.name is BROWSER_NAME.FIREFOX + is_opera: -> + return platform.name is BROWSER_NAME.OPERA + is_electron: -> + return navigator.userAgent.includes BROWSER_NAME.ELECTRON + + get_version: -> + return window.parseInt platform.version?.split('.')[0], 10 + + requires_codec_rewrite: -> + return false if not @supports_calling() + return @is_chrome() and @get_version() in [50, 51] + + supports_notifications: -> + return false if window.Notification is undefined + return false if window.Notification.requestPermission is undefined + return false if document.visibilityState is undefined + return true + supports_calling: -> + return false if window.WebSocket is undefined + return false if not navigator.mediaDevices?.getUserMedia + return false if @is_edge() + return @is_chrome() or @is_firefox() or @is_opera() + supports_screen_sharing: -> + return true if window.desktopCapturer + return @is_firefox() and @get_version() >= 48 + + os = + is_mac: -> + return navigator.platform.includes PLATFORM_NAME.MACINTOSH + is_windows: -> + return navigator.platform.includes PLATFORM_NAME.WINDOWS + + # add body information + os_css_class = if os.is_mac() then 'os-mac' else 'os-pc' + platform_css_class = if _check.is_electron() then 'platform-electron' else 'platform-web' + $(document.body).addClass "#{os_css_class} #{platform_css_class}" + + app_version = -> + if $("[property='wire:version']").attr('version')? + version = $("[property='wire:version']").attr('version').trim().split '-' + return "#{version[0]}.#{version[1]}.#{version[2]}.#{version[3]}#{version[4]}" + + ################ + # PUBLIC METHODS + ################ + + # "backend.current" is "undefined" when you are not connected to the backend (for example, if you are on the login page). + # In such situations use methods like "is_staging" to detect environments. + # + backend: current: undefined + + frontend: + is_localhost: -> + return window.location.hostname in [APP_ENV.LOCALHOST, APP_ENV.VIRTUAL_HOST] + is_production: -> + return window.location.hostname in [APP_ENV.PRODUCTION, APP_ENV.PROD_NEXT, APP_ENV.TACO] + + browser: + name: platform.name + version: _check.get_version() + + chrome: _check.is_chrome() + edge: _check.is_edge() + firefox: _check.is_firefox() + opera: _check.is_opera() + + supports: + calling: _check.supports_calling() + notifications: _check.supports_notifications() + screen_sharing: _check.supports_screen_sharing() + requires: + calling_codec_rewrite: _check.requires_codec_rewrite() + + os: + linux: not os.is_mac() and not os.is_windows() + mac: os.is_mac() + win: os.is_windows() + + electron: _check.is_electron() + + version: (show_wrapper_version = true) -> + return 'dev' if z.util.Environment.frontend.is_localhost() + return window.electron_version if window.electron_version and show_wrapper_version + return app_version() diff --git a/app/script/util/Invite.coffee b/app/script/util/Invite.coffee new file mode 100644 index 00000000000..30085bc56a7 --- /dev/null +++ b/app/script/util/Invite.coffee @@ -0,0 +1,162 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +### + Encoding + ========= + + The initial data is: + +-----------------------+--------------+--------------------------+ + | 14 bytes random data | 2 bytes time | 16 bytes UUID | + +-----------------------+--------------+--------------------------+ + + The initial data is then encoded with AES, no IV, no padding. + + Time is the number of hours since 01 Jan 2014 00:00 as a Big Endian unsigned int16. +### + +z.util.Invite = do -> + + RANDOM_DATA_SIZE = 14 + TIME_DATA_SIZE = 2 + UUID_DATA_SIZE = 16 + BUFFER_SIZE = RANDOM_DATA_SIZE + TIME_DATA_SIZE + UUID_DATA_SIZE + + HOUR_IN_SEC = 60 * 60 + + REFERENCE_TIMESTAMP = 1388534400 # 01 Jan 2014 00:00:00 + + INVITATION_TO_CONNECT_BASE_URL = 'https://wire.com/c/' + INVITATION_TO_CONNECT_TTL = 14 # 2 weeks + + # AES + KEY = new Uint8Array [ + 0x64, 0x68, 0xca, 0xee, 0x5c, 0x0, 0x25, 0xf5, 0x68, 0xe4, 0xd0, 0x85, 0xf8, 0x38, 0x28, 0x6a, + 0x8a, 0x98, 0x6d, 0x2d, 0xfa, 0x67, 0x5e, 0x48, 0xa3, 0xed, 0x2a, 0xef, 0xdd, 0xaf, 0xe8, 0xc1 + ] + + ### + Get difference between date and reference date in hours + + @param date [Date] + ### + hours_between_reference_date_and_date: (date) -> + diff = date.getTime() / 1000 - REFERENCE_TIMESTAMP + return Math.floor diff / HOUR_IN_SEC + + ### + Generate invitation token based on user id and expiration date + + @param user_id [String] + @param expiration_date [Date] + ### + encode_data: (user_id, expiration_date) -> + decrypted_data = new Uint8Array BUFFER_SIZE + + # 14 bytes random data + for i in [0...RANDOM_DATA_SIZE] + decrypted_data[i] = (Math.random() * 0x100000000) | 0 + + # 2 bytes time + hours = @hours_between_reference_date_and_date expiration_date + hours_left_byte = hours >>> 8 + hours_right_byte = hours & 255 + decrypted_data.set [hours_left_byte, hours_right_byte], RANDOM_DATA_SIZE + # 16 bytes UUID + uuid_bytes = z.util.uuid_to_bytes user_id + decrypted_data.set uuid_bytes, RANDOM_DATA_SIZE + TIME_DATA_SIZE + + # encrypt without initial vector and padding + encrypted_data = CryptoJS.AES.encrypt CryptoJS.lib.WordArray.create(decrypted_data), CryptoJS.lib.WordArray.create(KEY), + iv: new CryptoJS.lib.WordArray.init(), + padding: CryptoJS.pad.NoPadding + + encrypted_data_base64 = encrypted_data.toString() + + # remove padding and make base64 string url safe + encrypted_data_base64 = encrypted_data_base64 + .replace /\=$/, '' + .replace /\//g, '_' + .replace /\+/g, '-' + + return encrypted_data_base64 + + ### + Extract user id and expiration_date from given invitation token + + @param user_id [String] + @param expiration_date [Date] + ### + decode_data: (data) -> + # add padding and revert url safe characters + data = data + .concat '=' + .replace /_/g, '/' + .replace /-/g, '+' + + # decrypt + decrypted_data = CryptoJS.AES.decrypt data, CryptoJS.lib.WordArray.create(KEY), + iv: new CryptoJS.lib.WordArray.init(), + padding: CryptoJS.pad.NoPadding + + decrypted_data_bytes = z.util.base64_to_array decrypted_data.toString(CryptoJS.enc.Base64) + + return if not decrypted_data_bytes + + # no typed array slice in all browsers + decrypted_data_bytes = Array.prototype.slice.call decrypted_data_bytes + + # decode user id + uuid_bytes = decrypted_data_bytes.slice 16, decrypted_data_bytes.length + uuid = z.util.bytes_to_uuid uuid_bytes + + # decode timestamp + timestamp_bytes = decrypted_data_bytes.slice 14, 16 + timestamp = timestamp_bytes[0] << 8 | timestamp_bytes[1] & 255 + + return [uuid, timestamp] + + ### + Generate invite url for given user id + + @param user_id [String] receiver will be connected with this user + ### + get_invitation_to_connect_url: (user_id) -> + expiration_date = new Date() + expiration_date.setDate expiration_date.getDate() + INVITATION_TO_CONNECT_TTL + return INVITATION_TO_CONNECT_BASE_URL + @encode_data user_id, expiration_date + + ### + Extract user id from invite token. Will return undefined if the token is expired. + + @param token [String] invite to connect token + ### + get_user_from_invitation_token: (token) -> + [uuid, timestamp] = @decode_data token + current_timestamp = @hours_between_reference_date_and_date new Date() + is_valid_token = current_timestamp <= timestamp + return uuid if is_valid_token + return null + + ### + Extract user id from invite url. Will return undefined if the token is expired. + + @param url [String] invite to connect url + ### + get_user_from_invitation_url: (url) -> + return @get_user_from_invitation_token url.replace INVITATION_TO_CONNECT_BASE_URL, '' diff --git a/app/script/util/Statistics.coffee b/app/script/util/Statistics.coffee new file mode 100644 index 00000000000..933345867e2 --- /dev/null +++ b/app/script/util/Statistics.coffee @@ -0,0 +1,60 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} +z.util.Statistics ?= {} + +### +Calculates the average of all values within an array. + +@param values [Array] Input values +@param sum [Integer] (optional) Sum value + +@return [Integer] Average value +### +z.util.Statistics.average = (values) -> + return (z.util.Statistics.sum(values) / values.length).toFixed 2 + + +### +Calculates the sum of all value within an array. + +@param values [Array] Input values + +@return [Integer] Sum value +### +z.util.Statistics.sum = (values) -> + return values.reduce (sum, value) -> + sum + value + , 0 + + +### +Calculates the standard deviation within an array. + +@param values [Array] Input values +@param average [Integer] (optional) Average value +### +z.util.Statistics.standard_deviation = (values, average) -> + average = z.util.Statistics.average values if not average + squared_deviations = values.map (value) -> + deviation = value - average + return deviation * deviation + + return (Math.sqrt z.util.Statistics.average squared_deviations).toFixed 2 diff --git a/app/script/util/Time.coffee b/app/script/util/Time.coffee new file mode 100644 index 00000000000..d1220ff4467 --- /dev/null +++ b/app/script/util/Time.coffee @@ -0,0 +1,42 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +z.util.Time = do -> + ISO8601 = {} + + ISO8601.get_time_difference_in_millis = (iso_string_a, iso_string_b) -> + date_a_in_millis = new Date(iso_string_a).getTime() + date_b_in_millis = new Date(iso_string_b).getTime() + difference = date_a_in_millis - date_b_in_millis + return undefined if window.isNaN difference + return Math.abs(date_a_in_millis - date_b_in_millis) + + ISO8601.get_time_difference_in_seconds = (iso_string_a, iso_string_b) -> + difference = ISO8601.get_time_difference_in_millis iso_string_a, iso_string_b + return undefined if window.isNaN difference + return window.parseInt Math.abs(difference / 1000), 10 + + ISO8601.get_time_difference_in_minutes = (iso_string_a, iso_string_b) -> + difference = ISO8601.get_time_difference_in_millis iso_string_a, iso_string_b + return undefined if window.isNaN difference + return Math.abs(((difference) / 1000) / 60) + + public_methods = + ISO8601: ISO8601 + + return public_methods diff --git a/app/script/util/TimeTracker.coffee b/app/script/util/TimeTracker.coffee new file mode 100644 index 00000000000..4cd3c10a4c7 --- /dev/null +++ b/app/script/util/TimeTracker.coffee @@ -0,0 +1,49 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +class z.util.TimeTracker + constructor: -> + @time = + start: -1 + stop: -1 + tracked: -1 + in_seconds: -1 + in_mseconds: -1 + + start_timer: => + @time.start = (window.performance or Date).now() + + stop_timer: => + if @time.start is -1 + # You have to start the timer before logging. + return + + @time.stop = (window.performance or Date).now() + @time.tracked = @time.stop - @time.start + @time.in_mseconds = Math.round @time.tracked + @time.in_seconds = (@time.tracked / 1000).toFixed 2 + @get_formatted_time_message() + + get_formatted_time: => + return "#{@time.in_mseconds}ms (#{@time.in_seconds}s)" + + get_formatted_time_message: => + return "Time measured was #{@get_formatted_time()}." diff --git a/app/script/util/confirm.coffee b/app/script/util/confirm.coffee new file mode 100644 index 00000000000..25ba95c21c0 --- /dev/null +++ b/app/script/util/confirm.coffee @@ -0,0 +1,80 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + + + +# show confirm template inside the specified element +# +# @example show confirm dialog +# +# $('#parent').confirm +# template: '#template-confirm' +# data: +# foo: 'bar' +# confirm: -> +# ... do something on confirm ... +# cancel: -> +# ... do something on cancel ... +# +# @param [Object] +# @option template: [String] template that will be displayed +# @option data: [object] used as viewmodel for this dialog +# @option confirm: [Function] will be executed when confirm button is clicked +# @option cancel: [Function] will be executed when cancel button is clicked +# +$.fn.confirm = (config) -> + + template_html = $(config.template).html() + parent = $(@) + parent.append template_html + confirm = parent.find '.confirm' + group = parent.find '.participants-group' + is_visible = true + + is_small = group.hasClass 'small' + group.removeClass 'small' if is_small + + ko.applyBindings config.data, confirm[0] + + if config.data?.user? + stripped_user_name = config.data.user.first_name() + parent + .find '.user' + .html z.util.escape_html stripped_user_name + + requestAnimFrame -> + confirm.addClass 'confirm-is-visible' + + @destroy = -> + is_visible = false + ko.cleanNode confirm[0] + group.addClass 'small' if is_small + parent.find('.confirm').remove() + + @is_visible = -> + return is_visible + + $('[data-action="cancel"]', confirm).click => + config.cancel? config.data + @destroy() + + $('[data-action="confirm"]', confirm).click => + config.confirm? config.data + @destroy() + + return @ diff --git a/app/script/util/keycode.coffee b/app/script/util/keycode.coffee new file mode 100644 index 00000000000..710efc1ea96 --- /dev/null +++ b/app/script/util/keycode.coffee @@ -0,0 +1,31 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z = window.z or {} +z.util = z.util or {} + +z.util.KEYCODE = + ARROW_DOWN: 40 + ARROW_LEFT: 37 + ARROW_RIGHT: 39 + ARROW_UP: 38 + BACKSPACE: 46 + DELETE: 8 + ESC: 27 + ENTER: 13 + V: 86 diff --git a/app/script/util/protobuf.coffee b/app/script/util/protobuf.coffee new file mode 100644 index 00000000000..7da0c62b6df --- /dev/null +++ b/app/script/util/protobuf.coffee @@ -0,0 +1,32 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +z.util.protobuf = + + load_protos: (file) -> + return new Promise (resolve, reject) -> + dcodeIO.ProtoBuf.loadProtoFile file, (error, builder) -> + if error + reject new Error "Loading protocol buffer file failed: #{error.message}" + else + z.proto ?= {} + _.extend z.proto, builder.build() + resolve() diff --git a/app/script/util/scroll-helpers.coffee b/app/script/util/scroll-helpers.coffee new file mode 100644 index 00000000000..9e72bab960e --- /dev/null +++ b/app/script/util/scroll-helpers.coffee @@ -0,0 +1,58 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + + + +$.fn.scroll_end = -> + element = $(@).get 0 + return if not element + return element.scrollHeight - element.clientHeight + + +$.fn.scroll_to_bottom = -> + $element = $(@) + return if $element.length is 0 + $element.scrollTop $element[0].scrollHeight + window.setTimeout => + $element.scrollTop $element[0].scrollHeight if not $(@).is_scrolled_bottom() + , 200 + +$.fn.scroll_by = (distance) -> + $element = $(@) + return if $element.length is 0 + scroll_top = $element[0].scrollTop + $element.scrollTop scroll_top + distance + + +$.fn.is_scrolled_bottom = (offset = 0) -> + $element = $(@) + return if $element.length is 0 + scroll_top = Math.ceil $element.scrollTop() + scroll_height = $element[0].scrollHeight + height = $element[0].clientHeight + return scroll_top + height + offset >= scroll_height + +$.fn.is_scrolled_top = -> + $element = $(@) + return if $element.length is 0 + return $element.scrollTop() is 0 + +$.fn.is_scrollable = -> + element = $(@).get 0 + return if not element + return element.scrollHeight > element.clientHeight diff --git a/app/script/util/shims.coffee b/app/script/util/shims.coffee new file mode 100644 index 00000000000..54cf4407da7 --- /dev/null +++ b/app/script/util/shims.coffee @@ -0,0 +1,29 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} + +# http://www.paulirish.com/2011/requestanimationframe-for-smart-animating/ +window.requestAnimFrame = (-> + window.requestAnimationFrame or + window.webkitRequestAnimationFrame or + window.mozRequestAnimationFrame or + (callback) -> + window.setTimeout callback, 1000 / 60 +)() diff --git a/app/script/util/storage.coffee b/app/script/util/storage.coffee new file mode 100644 index 00000000000..a9d6b0f927a --- /dev/null +++ b/app/script/util/storage.coffee @@ -0,0 +1,39 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.storage ?= {} + +### +# BIG TODO: Turn this into a "z.util.StorageUtil" because the "z.storage" namespace is already occupied! +### + +z.storage.get_value = (key) -> + amplify.store key + + +z.storage.reset_value = (key) -> + z.storage.set_value key, null + + +z.storage.set_value = (key, value, seconds_to_expire) -> + if seconds_to_expire + amplify.store key, value, + expires: seconds_to_expire * 1000 + else + amplify.store key, value diff --git a/app/script/util/types.coffee b/app/script/util/types.coffee new file mode 100644 index 00000000000..eec5f7ac906 --- /dev/null +++ b/app/script/util/types.coffee @@ -0,0 +1,31 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.util ?= {} +z.util.types ?= {} + +z.util.types.convert_array_buffer_to_string = (input) -> + if _.isString input + payload = input + else + uncompressed = pako.inflate input + payload = '' + for char, i in uncompressed + payload = "#{payload}#{String.fromCharCode uncompressed[i]}" + return payload diff --git a/app/script/util/util.coffee b/app/script/util/util.coffee new file mode 100644 index 00000000000..16c674d9a0f --- /dev/null +++ b/app/script/util/util.coffee @@ -0,0 +1,685 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# +# This module of the Wire Software uses software code from Nicholas Fisher +# governed by the MIT license (https://github.com/KyleAMathews/deepmerge). +# +## The MIT License (MIT) +## +## Copyright (c) 2012 Nicholas Fisher +## +## Permission is hereby granted, free of charge, to any person obtaining a copy +## of this software and associated documentation files (the "Software"), to deal +## in the Software without restriction, including without limitation the rights +## to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +## copies of the Software, and to permit persons to whom the Software is +## furnished to do so, subject to the following conditions: +## +## The above copyright notice and this permission notice shall be included in all +## copies or substantial portions of the Software. +## +## THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +## IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +## FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +## AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +## LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +## OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +## SOFTWARE. + +window.z ?= {} +z.util ?= {} + +window.LOG = -> + console?.log? arguments... + + +z.util.dummy_image = (width, height) -> + canvas = document.createElement 'canvas' + canvas.width = width + canvas.height = height + ctx = canvas.getContext '2d' + ctx.fillStyle = '#fff' + ctx.fillRect 0, 0, width, height + return canvas.toDataURL 'image/png' + + +z.util.is_same_location = (past_location, current_location) -> + return past_location isnt '' and current_location.startsWith past_location + + +z.util.iterate_array_index = (array, current_index) -> + return undefined if not _.isArray(array) or not _.isNumber current_index + return undefined if not array.length + return (current_index + 1) % array.length + + +z.util.load_file_buffer = (file) -> + return new Promise (resolve, reject) -> + reader = new FileReader() + reader.onload = -> resolve @result + reader.onerror = reject + reader.readAsArrayBuffer file + + +z.util.load_url_buffer = (url, xhr_accessor_function) -> + return new Promise (resolve, reject) -> + xhr = new XMLHttpRequest() + xhr.open 'GET', url, true + xhr.responseType = 'arraybuffer' + xhr.onload = -> resolve [xhr.response, xhr.getResponseHeader 'content-type'] + xhr.onerror = reject + xhr.onabort = reject + xhr_accessor_function? xhr + xhr.send() + + +z.util.load_url_blob = (url, callback) -> + z.util.load_url_buffer url + .then (value) -> + [buffer, type] = value + image_as_blob = new Blob [new Uint8Array buffer], type: type + callback? image_as_blob + + +z.util.append_url_parameter = (url, param) -> + separator = if z.util.contains url, '?' then '&' else '?' + return "#{url}#{separator}#{param}" + + +z.util.get_url_parameter = (name) -> + params = window.location.search.substring(1).split '&' + for param in params + value = param.split '=' + return unescape value[1] if value[0] is name + return null + + +### +Get extension of a filename. + +@param filename [String] filename including extension +@return [String] +### +z.util.get_file_extension = (filename) -> + return '' if not filename.includes '.' + return 'tar.gz' if filename.includes 'tar.gz' + return filename.substr filename.lastIndexOf('.') + 1 + +### +Remove extension of a filename. + +@param filename [String] filename including extension +@return [String] New String without extension +### +z.util.trim_file_extension = (filename) -> + filename = filename.replace '.tar.gz', '' + return filename.replace /\.[^/.]+$/, '' + +### +Format bytes into a human readable string. + +@param bytes [Number] bytes to format +@param decimals [Number] Number of decimals to keep +@return [String] +### +z.util.format_bytes = (bytes, decimals) -> + return '0B' if bytes is 0 + kilobytes = 1024 + decimals = decimals + 1 or 2 + unit = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'] + index = Math.floor Math.log(bytes) / Math.log(kilobytes) + return parseFloat((bytes / Math.pow(kilobytes, index)).toFixed(decimals)) + unit[index] + +### +Format seconds into hh:mm:ss. + +@param bytes [seconds] duration to format in seconds +@return [String] +### +z.util.format_seconds = (duration) -> + duration = Math.round duration or 0 + + hours = Math.floor duration / (60 * 60) + divisor_for_minutes = duration % (60 * 60) + minutes = Math.floor divisor_for_minutes / 60 + divisor_for_seconds = divisor_for_minutes % 60 + seconds = Math.ceil divisor_for_seconds + + components = [ + z.util.zero_padding minutes + z.util.zero_padding seconds + ] + + if hours > 0 + components.unshift hours + + return components.join ':' + + +z.util.get_content_type_from_data_url = (data_url) -> + return data_url.split(',')[0].split(':')[1].split(';')[0] + + + +z.util.strip_data_uri = (string) -> + return string.replace /^data:.*,/, '' + + +### +Convert base64 string to UInt8Array. +Function will remove data uri if present + +@param base64 [String] base64 encoded string +@return [UInt8Array] +### +z.util.base64_to_array = (base64) -> + return sodium.from_base64 z.util.strip_data_uri base64 + +### +Convert ArrayBuffer or UInt8Array to base64 string + +@param array [ArrayBuffer|UInt8Array] raw binary data or bytes +@return [String] base64 encoded string +### +z.util.array_to_base64 = (array) -> + return sodium.to_base64 new Uint8Array(array), true + +### +Convert base64 dataURI to Blob + +@param base64 [String] base64 encoded data uri +@return [Blob] +### +z.util.base64_to_blob = (base64) -> + mime_type = z.util.get_content_type_from_data_url base64 + bytes = z.util.base64_to_array base64 + return new Blob [bytes], 'type': mime_type + +z.util.encode_base64_md5_array_buffer_view = (array_buffer_view) -> + word_array = CryptoJS.lib.WordArray.create array_buffer_view + md5_hash = CryptoJS.MD5(word_array).toString() + md5_hash_hex = z.util.read_string_chars_as_hex md5_hash + return btoa md5_hash_hex + +z.util.download_text = (content) -> + blob = new Blob([content], 'type': 'text/plain') + z.util.download_blob blob, 'download.txt' + + +### +Downloads blob using a hidden link element. + +@param blob [Blob] Blob to store +@param filename [String] Data will be saved under this name +### +z.util.download_blob = (blob, filename) -> + url = window.URL.createObjectURL blob + link = document.createElement 'a' + document.body.appendChild link + link.href = url + link.download = filename + link.style = 'display: none' + link.click() + + # Wait before removing resource and link. Needed in FF + setTimeout -> + document.body.removeChild link + window.URL.revokeObjectURL url + , 100 + + +z.util.read_deferred = (file, type) -> + deferred = new $.Deferred() + reader = new FileReader() + reader.onload = (e) -> deferred.resolve e.target.result + reader.onerror = (e) -> deferred.reject @ + + if type is 'buffer' + reader.readAsArrayBuffer file + else if type is 'url' + reader.readAsDataURL file + else + reader.readAsText file + + return deferred.promise() + + +z.util.phone_uri_to_e164 = (phone_number) -> + return phone_number.replace /tel:|-/g, '' + + +z.util.phone_number_to_e164 = (phone_number, country_code) -> + return window.PhoneFormat.formatE164 "#{country_code}".toUpperCase(), "#{phone_number}" + + +z.util.create_random_uuid = -> + return UUID.genV4().hexString + + +z.util.bytes_to_uuid = (bytes) -> + hex = [] + i = 0 + while i < bytes.length + hex.push (bytes[i] >>> 4).toString 16 + hex.push (bytes[i] & 0xF).toString 16 + i++ + hex.join('').replace /(\w{8})(\w{4})(\w{4})(\w{4})(\w{12})/, '$1-$2-$3-$4-$5' + + +z.util.uuid_to_bytes = (hex) -> + parts = hex.split '-' + ints = [] + intPos = 0 + i = 0 + + while i < parts.length + j = 0 + + while j < parts[i].length + ints[intPos++] = window.parseInt parts[i].substr(j, 2), 16 + j += 2 + i++ + ints + + +z.util.encode_base64 = (text) -> + return window.btoa text + + +z.util.encode_sha256_base64 = (text) -> + return CryptoJS.SHA256(text).toString CryptoJS.enc.Base64 + + +z.util.escape_html = (html) -> + return _.escape html + + +z.util.alias = + # Note IE10 listens to "transitionend" instead of "animationend" + animationend: 'transitionend animationend oAnimationEnd MSAnimationEnd mozAnimationEnd webkitAnimationEnd' + + +z.util.add_blank_targets = (text_with_anchors) -> + return "#{text_with_anchors}".replace /rel="nofollow"/gi, 'target="_blank" rel="nofollow"' + + +z.util.auto_link_emails = (text) -> + email_pattern = /([a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9-.]+)/gim + return text.replace email_pattern, '$1' + + +z.util.get_last_characters = (message, amount) -> + return false if message.length < amount + return message.substring message.length - amount + + +z.util.cut_last_characters = (message, amount) -> + return message.substring 0, message.length - amount + + +z.util.markup_links = (message) -> + return message.replace(/ + destination[key] = target[key] + + Object.keys(source).forEach (key) -> + if typeof source[key] isnt 'object' or not source[key] + destination[key] = source[key] + else + if not target[key] + destination[key] = source[key] + else + destination[key] = z.util.merge_objects target[key], source[key] + + return destination + +# Note: We are using "Underscore.js" to escape HTML in the original message +z.util.render_message = (message, theme_color) -> + message = marked message + message = z.util.auto_link_emails message + message = message.replace /\n/g, '
    ' + # Remove
    if it is the last thing in a message + if z.util.get_last_characters(message, '
    '.length) is '
    ' + message = z.util.cut_last_characters message, '
    '.length + + return z.media.MediaParser.render_media_embeds message, theme_color + + +z.util.read_string_chars_as_hex = (text) -> + text.match(/../g).map (x) -> + String.fromCharCode window.parseInt x, 16 + .join '' + + +# append array to knockout observableArray +# source: https://github.com/knockout/knockout/issues/416 +z.util.ko_array_push_all = (ko_array, values_to_push) -> + underlyingArray = ko_array() + ko_array.valueWillMutate() + ko.utils.arrayPushAll underlyingArray, values_to_push + ko_array.valueHasMutated() + + +# prepend array to knockout observableArray +z.util.ko_array_unshift_all = (ko_array, values_to_shift) -> + underlyingArray = ko_array() + ko_array.valueWillMutate() + Array.prototype.unshift.apply underlyingArray, values_to_shift + ko_array.valueHasMutated() + + +### +Add zero padding until limit is reached + +@param value [String|Number] +@return [String] +### +z.util.zero_padding = (value, length = 2) -> + if value.toString().length < length + return z.util.zero_padding "0#{value}", length + return "#{value}" + + +### +Human readable format of a timestamp. Not testable due to timezones :( + +@param timestamp [Number] +@return [String] +### +z.util.format_timestamp = (timestamp) -> + datetime = new Date timestamp + + year = datetime.getFullYear() + month = z.util.zero_padding datetime.getMonth() + 1 + day = z.util.zero_padding datetime.getDate() + hours = z.util.zero_padding datetime.getHours() + minutes = z.util.zero_padding datetime.getMinutes() + seconds = z.util.zero_padding datetime.getSeconds() + + return "#{day}.#{month}.#{year} (#{hours}:#{minutes}:#{seconds})" + + +z.util.sort_groups_by_last_event = (group_a, group_b) -> + return group_b.last_event_timestamp() - group_a.last_event_timestamp() + +### + Returns a copy of an object, which is ordered by the keys of the original object. +### +z.util.sort_object_by_keys = (object, reverse) -> + sorted_object = {} + + keys = Object.keys object + keys.sort() + + if reverse + for key in keys by -1 + value = object[key] + sorted_object[key] = value + else + for key in keys + value = object[key] + sorted_object[key] = value + + return sorted_object + +z.util.sort_user_by_first_name = (user_a, user_b) -> + name_a = user_a.first_name().toLowerCase() + name_b = user_b.first_name().toLowerCase() + return -1 if name_a < name_b + return 1 if name_a > name_b + return 0 + + +z.util.sort_user_by_name = (user_a, user_b) -> + name_a = user_a.name().toLowerCase() + name_b = user_b.name().toLowerCase() + return -1 if name_a < name_b + return 1 if name_a > name_b + return 0 + + +z.util.remove_line_breaks = (string) -> + string.replace /(\r\n|\n|\r)/gm, '' + + +z.util.trim_line_breaks = (string) -> + string.replace /^\s+|\s+$/g, '' + + +z.util.contains = (string = '', query) -> + string = string.toLowerCase() + query = query.toLowerCase() + return string.indexOf(query) isnt -1 + + +z.util.array_get_next = (array, item, filter) -> + index = array.indexOf item + next_index = index + 1 + + # couldn't find the item + return null if index is -1 + + # item is last item in the array + return array[index - 1] if next_index is array.length and index > 0 + + return if next_index >= array.length + + for i in [next_index..array.length] + current_item = array[i] + return current_item unless filter? and not filter current_item + + +z.util.array_is_last = (array, item) -> + return array.indexOf(item) is array.length - 1 + + +# returns chunks of the given size +z.util.array_chunks = (array, size) -> + chunks = [] + temp_array = [].concat array + while temp_array.length + chunks.push temp_array.splice 0, size + return chunks + + +# This will remove url(' and url(" from the beginning of the string. +# It will also remove ") and ') from the end if present. +z.util.strip_url_wrapper = (url) -> + return url.replace(/^url\(["']?/, '').replace /["']?\)$/, '' + + +z.util.valid_profile_image_size = (file, min_width, min_height, callback) -> + image = new Image() + image.onload = -> + callback image.width >= min_width and image.height >= min_height + image.src = window.URL.createObjectURL file + + +z.util.trunc_text = (text, output_length, word_boundary = true) -> + if text.length > output_length + trunc_index = output_length - 1 + if word_boundary and text.lastIndexOf(' ', output_length - 1) > output_length - 25 + trunc_index = text.lastIndexOf ' ', output_length - 1 + text = "#{text.substr 0, trunc_index}#{z.localization.Localizer.get_text z.string.truncation}" + return text + + +z.util.is_valid_email = (email) -> + re = /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/ + return re.test email + + +### +Checks if input has the format of an international phone number + +@note Begins with + and contains only numbers + +@param phone_number [String] Input value +@return [Boolean] Is the input a phone number string +### +z.util.is_valid_phone_number = (phone_number) -> + re = /^\+?[1-9]\d{1,14}$/ + return re.test phone_number + + +z.util.get_first_character = (string) -> + re = new RegExp /([\uE000-\uF8FF]|\uD83C[\uDF00-\uDFFF]|\uD83D[\uDC00-\uDDFF])/ + find_emoji_in_string = re.exec string + if find_emoji_in_string and find_emoji_in_string.index is 0 + return find_emoji_in_string[0] + else + return string[0] + + +z.util.string_format = -> + s = arguments[0] + for i in [0...arguments.length] + reg = new RegExp "\\{#{i}\\}", 'gm' + s = s.replace reg, arguments[++i] + return s + +### +JS Implementation of MurmurHash3 (r136) (as of May 20, 2011) + +@param key [String] ASCII only +@param seed [Integer] Positive integer only + +@return [Integer] 32-bit positive integer +### +z.util.murmurhash3 = `function(key, seed){ + var remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i; + + remainder = key.length & 3; // key.length % 4 + bytes = key.length - remainder; + h1 = seed; + c1 = 0xcc9e2d51; + c2 = 0x1b873593; + i = 0; + + while (i < bytes) { + k1 = + ((key.charCodeAt(i) & 0xff)) | + ((key.charCodeAt(++i) & 0xff) << 8) | + ((key.charCodeAt(++i) & 0xff) << 16) | + ((key.charCodeAt(++i) & 0xff) << 24); + ++i; + + k1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff; + k1 = (k1 << 15) | (k1 >>> 17); + k1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff; + + h1 ^= k1; + h1 = (h1 << 13) | (h1 >>> 19); + h1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff; + h1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16)); + } + + k1 = 0; + + switch (remainder) { + case 3: + k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16; + case 2: + k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8; + case 1: + k1 ^= (key.charCodeAt(i) & 0xff); + + k1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff; + k1 = (k1 << 15) | (k1 >>> 17); + k1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff; + h1 ^= k1; + } + + h1 ^= key.length; + + h1 ^= h1 >>> 16; + h1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff; + h1 ^= h1 >>> 13; + h1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff; + h1 ^= h1 >>> 16; + + return h1 >>> 0; +}` + +z.util.get_unix_timestamp = -> + return Math.floor Date.now() / 1000 + + +z.util.get_first_name = (user_et, declension = z.string.Declension.NOMINATIVE) -> + if user_et.is_me + if declension is z.string.Declension.NOMINATIVE + return z.localization.Localizer.get_text z.string.conversation_you_nominative + else if declension is z.string.Declension.DATIVE + return z.localization.Localizer.get_text z.string.conversation_you_dative + else if declension is z.string.Declension.ACCUSATIVE + return z.localization.Localizer.get_text z.string.conversation_you_accusative + return user_et.first_name() + + +z.util.compare_names = (name_a, name_b) -> + z.util.contains window.getSlug(name_a), window.getSlug(name_b) + + +z.util.capitalize_first_char = (s) -> + return "#{s.charAt(0).toUpperCase()}#{s.substring 1}" + +z.util.print_devices_id = (id) -> + return '' if not id + id_with_padding = z.util.zero_padding id, 16 + prettified_id = '' + prettified_id += "#{part}" for part in id_with_padding.match /.{1,2}/g + return prettified_id + +### +Returns bucket for given value based on the specified bucket limits + +@example z.util.bucket_values(13, [0, 5, 10, 15, 20, 25]) will return '11-15') + +@param value [Number] Numeric value that +@param bucket_limits [Array] Array with numeric values that define the upper limit of the bucket + +@return [String] bucket +### +z.util.bucket_values = (value, bucket_limits) -> + return '0' if value is 0 + + for limit, i in bucket_limits + if value <= limit + previous_limit = bucket_limits[i - 1] + return "#{previous_limit+1}-#{limit}" + + last_limit = bucket_limits[bucket_limits.length - 1] + return "#{last_limit+1}-" diff --git a/app/script/view_model/ActionsViewModel.coffee b/app/script/view_model/ActionsViewModel.coffee new file mode 100644 index 00000000000..3b309de0d89 --- /dev/null +++ b/app/script/view_model/ActionsViewModel.coffee @@ -0,0 +1,125 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +class z.ViewModel.ActionsViewModel + constructor: (element_id, @conversation_repository, @user_repository, @conversation_list) -> + @logger = new z.util.Logger 'z.ViewModel.ActionsViewModel', z.config.LOGGER.OPTIONS + + @action_bubbles = {} + @selected_conversation = ko.observable() + @conversations_archived = @conversation_repository.conversations_archived + @conversations_unarchived = @conversation_repository.conversations_unarchived + + @archive_conversation_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_list_archive + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.ARCHIVE} + } + @notify_conversation_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_list_notify + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.SILENCE} + } + @silence_conversation_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_list_silence + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.SILENCE} + } + + # fix for older wrapper versions + @conversation_list.click_on_archive_action = @click_on_archive_action + @conversation_list.click_on_block_action = @click_on_block_action + @conversation_list.click_on_cancel_action = @click_on_cancel_action + @conversation_list.click_on_clear_action = @click_on_clear_action + @conversation_list.click_on_leave_action = @click_on_leave_action + @conversation_list.click_on_mute_action = @click_on_mute_action + @conversation_list.selected_conversation = @selected_conversation + + @_init_subscriptions() + + ko.applyBindings @, document.getElementById element_id + + _init_subscriptions: => + amplify.subscribe z.event.WebApp.SHORTCUT.ARCHIVE, @click_on_archive_action + amplify.subscribe z.event.WebApp.SHORTCUT.SILENCE, @click_on_mute_action + amplify.subscribe z.event.WebApp.ACTION.SHOW, @click_on_actions + + click_on_actions: (conversation_et, event) => + @selected_conversation conversation_et + + $('.conversation-list-item').removeClass 'hover' + list_element = $(event.currentTarget.parentNode.parentNode).addClass 'hover' + + if not @action_bubbles[conversation_et.id] + @action_bubbles[conversation_et.id] = new zeta.webapp.module.Bubble + host_selector: "##{$(event.currentTarget).attr 'id'}" + scroll_selector: '.conversation-list-items' + on_hide: => + list_element.removeClass 'hover' + @action_bubbles[conversation_et.id] = undefined + + @action_bubbles[conversation_et.id].toggle() + + event.stopPropagation() + + click_on_archive_action: => + @conversation_repository.archive_conversation @_click_on_action() + + click_on_block_action: => + conversation_et = @_click_on_action() + next_conversation_et = @conversation_repository.get_next_conversation conversation_et + user_et = conversation_et.participating_user_ets()[0] + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.BLOCK, + data: user_et.first_name() + action: => @user_repository.block_user user_et, -> + amplify.publish z.event.WebApp.CONVERSATION.SWITCH, conversation_et, next_conversation_et + + click_on_cancel_action: => + conversation_et = @_click_on_action() + next_conversation_et = @conversation_repository.get_next_conversation conversation_et + @user_repository.cancel_connection_request conversation_et.participating_user_ets()[0], next_conversation_et + + click_on_clear_action: => + conversation_et = @_click_on_action() + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CLEAR, + data: conversation_et.display_name() + conversation: conversation_et + action: (leave = false) => @conversation_repository.clear_conversation conversation_et, leave + + click_on_leave_action: => + conversation_et = @_click_on_action() + next_conversation_et = @conversation_repository.get_next_conversation conversation_et + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.LEAVE, + data: conversation_et.display_name() + action: => @conversation_repository.leave_conversation conversation_et, next_conversation_et + + click_on_mute_action: => + @conversation_repository.toggle_silence_conversation @_click_on_action() + + click_on_unarchive_action: => + @conversation_repository.unarchive_conversation @_click_on_action(), => + amplify.subscribe z.event.WebApp.ARCHIVE.CLOSE if @conversation_repository.conversations_archived().length is 0 + + _click_on_action: => + conversation_et = @selected_conversation() or @conversation_repository.active_conversation() + return if not conversation_et + amplify.publish z.event.WebApp.ARCHIVE.CLOSE if not conversation_et.is_archived() + @action_bubbles[conversation_et.id]?.hide() + @selected_conversation null + return conversation_et diff --git a/app/script/view_model/ArchiveViewModel.coffee b/app/script/view_model/ArchiveViewModel.coffee new file mode 100644 index 00000000000..485caf73e82 --- /dev/null +++ b/app/script/view_model/ArchiveViewModel.coffee @@ -0,0 +1,72 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +class z.ViewModel.ArchiveViewModel + constructor: (element_id, @conversation_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ArchiveViewModel', z.config.LOGGER.OPTIONS + + @conversations_archived = @conversation_repository.conversations_archived + @is_archive_visible = ko.observable() + + @should_update_scrollbar = (ko.computed => + return @is_archive_visible() + ).extend notify: 'always', rateLimit: 500 + + @_init_subscriptions() + + ko.applyBindings @, document.getElementById element_id + + _init_subscriptions: => + amplify.subscribe z.event.WebApp.ARCHIVE.SHOW, @open + amplify.subscribe z.event.WebApp.SEARCH.SHOW, @close + amplify.subscribe z.event.WebApp.ARCHIVE.CLOSE, @close + + click_on_actions: (conversation_et, event) -> + amplify.publish z.event.WebApp.ACTION.SHOW, conversation_et, event + + click_on_close_archive: -> + amplify.publish z.event.WebApp.ARCHIVE.CLOSE + + click_on_archived_conversation: (conversation_et) => + @conversation_repository.unarchive_conversation conversation_et + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + amplify.publish z.event.WebApp.ARCHIVE.CLOSE + + open: => + $(document).on 'keydown.show_archive', (event) -> + amplify.publish z.event.WebApp.ARCHIVE.CLOSE if event.keyCode is z.util.KEYCODE.ESC + @conversation_repository.update_conversations @conversation_repository.conversations_archived() + @is_archive_visible Date.now() + @_show() + + close: => + $(document).off 'keydown.show_archive' + @_hide() + +############################################################################### +# Archive animations +############################################################################### + _show: -> + $('#archive').addClass 'archive-is-visible' + + _hide: -> + $('#archive').removeClass 'archive-is-visible' diff --git a/app/script/view_model/AuthViewModel.coffee b/app/script/view_model/AuthViewModel.coffee new file mode 100644 index 00000000000..e3d249f7609 --- /dev/null +++ b/app/script/view_model/AuthViewModel.coffee @@ -0,0 +1,1311 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +TIMEOUT = + SHORT: 500 + LONG: 2000 + +### +View model for the auth page. + +@param element_id [] CSS class of the element where this view should be applied to (like "auth-page") +@param auth [z.main.Auth] Class that holds objects needed for app authentication +### +# @formatter:off +class z.ViewModel.AuthViewModel + constructor: (element_id, @auth) -> + @logger = new z.util.Logger 'z.ViewModel.AuthViewModel', z.config.LOGGER.OPTIONS + + @event_tracker = new z.tracking.EventTrackingRepository() + @user_service = new z.user.UserService @auth.client + + # Cryptography + @asset_service = new z.assets.AssetService @auth.client + # TODO: Don't operate with the service directly. Get a repository! + @storage_service = new z.storage.StorageService() + @storage_repository = new z.storage.StorageRepository @storage_service + + @user_mapper = new z.user.UserMapper @asset_service + user_service = new z.user.UserService @auth.client + @user_repository = new z.user.UserRepository user_service, @asset_service + + @cryptography_service = new z.cryptography.CryptographyRepository @auth.client + @cryptography_repository = new z.cryptography.CryptographyRepository @cryptography_service, @storage_repository + @client_service = new z.client.ClientService @auth.client, @storage_service + @client_repository = new z.client.ClientRepository @client_service, @cryptography_repository + + @notification_service = new z.event.NotificationService @auth.client, @storage_service + @web_socket_service = new z.event.WebSocketService @auth.client + @event_repository = new z.event.EventRepository @web_socket_service, @notification_service, @cryptography_repository, @user_repository + + @pending_server_request = ko.observable false + @disabled_by_animation = ko.observable false + + @get_wire = ko.observable false + @session_expired = ko.observable false + + @client_type = ko.observable z.client.ClientType.TEMPORARY + @country_code = ko.observable '' + @country = ko.observable '' + @phone_number = ko.observable '' + @email = ko.observable '' + @name = ko.observable '' + @password = ko.observable '' + @persist = ko.observable if z.util.Environment.electron then true else false + @persist.subscribe (is_persistent) => + if is_persistent then @client_type z.client.ClientType.PERMANENT else z.client.ClientType.TEMPORARY + + @self_user = ko.observable() + + # Manage devices + @remove_form_error = ko.observable false + @device_modal = undefined + @permanent_devices = ko.computed => + client for client in @client_repository.clients() when client.type is z.client.ClientType.PERMANENT + + @registration_context = z.auth.AuthView.REGISTRATION_CONTEXT.EMAIL + @prefilled_email = '' + + @code_digits = ko.observableArray [ + ko.observable '' + ko.observable '' + ko.observable '' + ko.observable '' + ko.observable '' + ko.observable '' + ] + @code = ko.computed => return (digit() for digit in @code_digits()).join('').substr 0, 6 + @code.subscribe (code) => + @_clear_errors() if code.length is 0 + @verify_code() if code.length is 6 + @phone_number_e164 = => return "#{@country_code()}#{@phone_number()}" + + @code_interval_id = undefined + + @code_expiration_timestamp = ko.observable 0 + @code_expiration_in = ko.observable '' + @code_expiration_timestamp.subscribe (timestamp) => + @code_expiration_in moment.unix(timestamp).fromNow() + @code_interval_id = setInterval => + if timestamp <= z.util.get_unix_timestamp() + clearInterval @code_interval_id + return @code_expiration_timestamp 0 + @code_expiration_in moment.unix(timestamp).fromNow() + , 20000 + + @validation_errors = ko.observableArray [] + @failed_validation_email = ko.observable false + @failed_validation_password = ko.observable false + @failed_validation_name = ko.observable false + @failed_validation_code = ko.observable false + @failed_validation_phone = ko.observable false + @failed_validation_terms = ko.observable false + @accepted_terms_of_use = ko.observable false + @accepted_terms_of_use.subscribe => @clear_error z.auth.AuthView.TYPE.TERMS + + @can_login_email = ko.computed => + return not @disabled_by_animation() and @email().length isnt 0 and @password().length isnt 0 + + @can_login_phone = ko.computed => + return not @disabled_by_animation() and @country_code().length > 1 and @phone_number().length isnt 0 + + @can_register = ko.computed => + return not @disabled_by_animation() and @email().length isnt 0 and @name().length isnt 0 and @password().length isnt 0 + + @can_resend_code = ko.computed => + return not @disabled_by_animation() and @code_expiration_timestamp() < z.util.get_unix_timestamp() + + @can_resend_registration = ko.computed => + return not @disabled_by_animation() and @email().length isnt 0 + + @can_resend_verification = ko.computed => + return not @disabled_by_animation() and @email().length isnt 0 + + @account_retry_text = ko.computed => + return z.localization.Localizer.get_text + id: z.string.auth_posted_retry + replace: {placeholder: '%email', content: @email()} + @account_resend_text = ko.computed => + return z.localization.Localizer.get_text + id: z.string.auth_posted_resend + replace: {placeholder: '%email', content: @email()} + @verify_code_text = ko.computed => + phone_number = PhoneFormat.formatNumberForMobileDialing('', @phone_number_e164()) or @phone_number_e164() + return z.localization.Localizer.get_text + id: z.string.auth_verify_code_description + replace: {placeholder: '%@number', content: phone_number} + @verify_code_timer_text = ko.computed => + return z.localization.Localizer.get_text + id: z.string.auth_verify_code_resend_timer + replace: {placeholder: '%expiration', content: @code_expiration_in()} + @verify_email_headline = ko.computed => + return z.localization.Localizer.get_text + id: z.string.auth_verify_email_headline + replace: {placeholder: '%name', content: @self_user()?.first_name()} + + @visible_section = ko.observable undefined + @visible_mode = ko.observable undefined + @visible_method = ko.observable undefined + + @account_mode = ko.observable undefined + @account_mode_login = ko.computed => + login_modes = [ + z.auth.AuthView.MODE.ACCOUNT_LOGIN + z.auth.AuthView.MODE.ACCOUNT_EMAIL + z.auth.AuthView.MODE.ACCOUNT_PHONE + ] + return @account_mode() in login_modes + + @posted_mode = ko.observable undefined + @posted_mode_offline = ko.computed => + return @posted_mode() is z.auth.AuthView.MODE.POSTED_OFFLINE + @posted_mode_pending = ko.computed => + return @posted_mode() is z.auth.AuthView.MODE.POSTED_PENDING + @posted_mode_resend = ko.computed => + return @posted_mode() is z.auth.AuthView.MODE.POSTED_RESEND + @posted_mode_retry = ko.computed => + return @posted_mode() is z.auth.AuthView.MODE.POSTED_RETRY + @posted_mode_verify = ko.computed => + return @posted_mode() is z.auth.AuthView.MODE.POSTED_VERIFY + + # Debugging + if window.location.hostname is 'localhost' + live_reload = document.createElement 'script' + live_reload.id = 'live_reload' + live_reload.src = 'http://localhost:32123/livereload.js' + document.body.appendChild live_reload + $('html').addClass 'development' + + ko.applyBindings @, document.getElementById element_id + + @show_initial_animation = false + + @_init_base() + @_track_app_launch() + $(".#{element_id}").show() + + _init_base: -> + if z.util.get_url_parameter z.auth.URLParameter.CONNECT + @get_wire true + @registration_context = z.auth.AuthView.REGISTRATION_CONTEXT.GENERIC_INVITE + else if z.util.get_url_parameter z.auth.URLParameter.EXPIRED + @session_expired true + + modes_to_block = [ + z.auth.AuthView.MODE.HISTORY + z.auth.AuthView.MODE.LIMIT + z.auth.AuthView.MODE.POSTED + z.auth.AuthView.MODE.POSTED_PENDING + z.auth.AuthView.MODE.POSTED_RETRY + z.auth.AuthView.MODE.POSTED_VERIFY + z.auth.AuthView.MODE.VERIFY_CODE + z.auth.AuthView.MODE.VERIFY_ADD_EMAIL + ] + + if invite = z.util.get_url_parameter z.auth.URLParameter.INVITE + @register_from_invite invite + else if @_has_no_hash() and z.storage.get_value z.storage.StorageKey.AUTH.SHOW_LOGIN + @_set_hash z.auth.AuthView.MODE.ACCOUNT_LOGIN + else if @_get_hash() in modes_to_block + @_set_hash() + else + @_on_hash_change() + + $(window) + .on 'hashchange', @_on_hash_change + .on 'dragover drop', -> false + + $("[id^='wire-login'], [id^='wire-mail'], [id^='wire-register'], [id^='wire-phone-code']").prevent_prefill() + + # Select country based on location of user IP + @country_code (z.util.CountryCodes.get_country_code($('[name=geoip]').attr 'country') or 1).toString() + @changed_country_code() + + + ############################################################################### + # Invitation Stuff + ############################################################################### + + register_from_invite: (invite) => + @auth.repository.retrieve_invite invite + .then (invite_info) => + @registration_context = z.auth.AuthView.REGISTRATION_CONTEXT.PERSONAL_INVITE + @name invite_info.name + if invite_info.email + @email invite_info.email + @prefilled_email = invite_info.email + else + @logger.log @logger.levels.WARN, 'Invite information does not contain an email address' + @_set_hash z.auth.AuthView.MODE.ACCOUNT_REGISTER + @_on_hash_change() + .catch (error) => + if error.label is z.service.BackendClientError::LABEL.INVALID_INVITATION_CODE + @logger.log @logger.levels.WARN, 'Invalid Invitation Code' + else + Raygun.send new Error('Invitation not found'), {invite_code: invite, error: error} + @_on_hash_change() + + + ############################################################################### + # Form actions + ############################################################################### + + # Register a new user account. + register: => + return if @pending_server_request() or not @_validate_input z.auth.AuthView.MODE.ACCOUNT_REGISTER + + @pending_server_request true + @persist true + payload = @_create_payload z.auth.AuthView.MODE.ACCOUNT_REGISTER + @auth.repository.register payload + .then => + @get_wire false + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.ENTERED_CREDENTIALS, + {outcome: 'success'} + + # Track if the user changed the pre-filled email + if @prefilled_email is @email() + @auth.repository.get_access_token().then @_account_verified + else + @_set_hash z.auth.AuthView.MODE.POSTED + @auth.repository.get_access_token().then @_wait_for_activate + @pending_server_request false + .catch (error) => @_on_register_error error + + # Sign in using an email login. + sign_in_email: => + return if @pending_server_request() or not @_validate_input z.auth.AuthView.MODE.ACCOUNT_EMAIL + + @pending_server_request true + payload = @_create_payload z.auth.AuthView.MODE.ACCOUNT_EMAIL + @auth.repository.login payload, @persist() + .then => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.LOGGED_IN, + { + context: z.auth.AuthView.MODE.ACCOUNT_EMAIL + remember_me: @persist() + } + @_authentication_successful() + .catch (error) => + @pending_server_request false + $('#wire-login-password').focus() + if navigator.onLine + if error.label is z.service.BackendClientError::LABEL.PENDING_ACTIVATION + return @_set_hash z.auth.AuthView.MODE.POSTED_PENDING + else if error.label + @_add_error z.string.auth_error_sign_in, [z.auth.AuthView.TYPE.EMAIL, z.auth.AuthView.TYPE.PASSWORD] + else + @_add_error z.string.auth_error_misc + else + @_add_error z.string.auth_error_offline + @_has_errors() + + # Sign in using a phone number. + sign_in_phone: => + return if @pending_server_request() or not @_validate_input z.auth.AuthView.MODE.ACCOUNT_PHONE + + @pending_server_request true + payload = @_create_payload z.auth.AuthView.MODE.ACCOUNT_PHONE + @auth.repository.request_login_code payload + .then (response) => + clearInterval @code_interval_id + if response.expires_in + @code_expiration_timestamp z.util.get_unix_timestamp() + response.expires_in + else if not response.label + @code_expiration_timestamp z.util.get_unix_timestamp() + z.config.LOGIN_CODE_EXPIRATION + @_set_hash z.auth.AuthView.MODE.VERIFY_CODE + @pending_server_request false + event = new z.tracking.event.PhoneVerification 'signIn', 'succeeded', undefined + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event.name, event.attributes + .catch (error) => + @pending_server_request false + if navigator.onLine + switch error.label + when z.service.BackendClientError::LABEL.PENDING_LOGIN + return _on_code_request_success error + when z.service.BackendClientError::LABEL.PASSWORD_EXISTS + @_add_error z.string.auth_error_misc + when z.service.BackendClientError::LABEL.INVALID_PHONE + @_add_error z.string.auth_error_phone_number_unknown, z.auth.AuthView.TYPE.PHONE + else + @_add_error z.string.auth_error_misc + else + @_add_error z.string.auth_error_offline + @_has_errors() + event = new z.tracking.event.PhoneVerification 'signIn', 'error', error?.label + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event.name, event.attributes + + # Verify the security code on phone number login. + verify_code: => + return if @pending_server_request() or not @_validate_code() + + @pending_server_request true + payload = @_create_payload z.auth.AuthView.MODE.VERIFY_CODE + @auth.repository.login payload, @persist() + .then => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.LOGGED_IN, + { + context: z.auth.AuthView.MODE.ACCOUNT_EMAIL + remember_me: @persist() + } + event = new z.tracking.event.PhoneVerification 'postLogin', 'succeeded', undefined + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event.name, event.attributes + @_authentication_successful() + .catch (error) => + if @validation_errors().length is 0 + @_add_error z.string.auth_error_code, z.auth.AuthView.TYPE.CODE + @_has_errors() + @pending_server_request false + event = new z.tracking.event.PhoneVerification 'postLogin', 'error', error.label + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event.name, event.attributes + + # Add an email on phone number login. + verify_add_email: => + return if @pending_server_request() or not @_validate_input z.auth.AuthView.MODE.VERIFY_ADD_EMAIL + + @pending_server_request true + @user_service.change_own_password @password() + .catch (error) => + @logger.log @logger.levels.WARN, 'Could not change user password', error + return error + .then (error) => + if not error? or error.code is z.service.BackendClientError::STATUS_CODE.FORBIDDEN + @user_service.change_own_email @email() + else + @pending_server_request false + return @_has_errors() + .then => + @pending_server_request false + @_wait_for_update() + @_set_hash z.auth.AuthView.MODE.POSTED_VERIFY + .catch (error) => + if error + switch error.label + when z.service.BackendClientError::LABEL.BLACKLISTED_EMAIL + @_add_error z.string.auth_error_email_forbidden, z.auth.AuthView.TYPE.EMAIL + when z.service.BackendClientError::LABEL.KEY_EXISTS + @_add_error z.string.auth_error_email_exists, z.auth.AuthView.TYPE.EMAIL + when z.service.BackendClientError::LABEL.INVALID_EMAIL + @_add_error z.string.auth_error_email_malformed, z.auth.AuthView.TYPE.EMAIL + else + @_add_error z.string.auth_error_email_malformed, z.auth.AuthView.TYPE.EMAIL + @_has_errors() + + ### + Create the backend call payload. + + @param mode [z.auth.AuthView.MODE] View state of the authentication page + @private + ### + _create_payload: (mode) => + email = @email().trim().toLowerCase() + + switch mode + when z.auth.AuthView.MODE.ACCOUNT_REGISTER + payload = + email: email + invitation_code: z.util.get_url_parameter z.auth.URLParameter.INVITE + label: @client_repository.construct_cookie_label email, @client_type() + label_key: @client_repository.construct_cookie_label_key email, @client_type() + locale: moment.locale() + name: @name().trim() + password: @password() + return payload + when z.auth.AuthView.MODE.ACCOUNT_EMAIL + payload = + email: email + label: @client_repository.construct_cookie_label email, @client_type() + label_key: @client_repository.construct_cookie_label_key email, @client_type() + password: @password() + return payload + when z.auth.AuthView.MODE.ACCOUNT_PHONE then return {} = + force: false + phone: @phone_number_e164() + when z.auth.AuthView.MODE.POSTED_RESEND then return {} = + email: email + when z.auth.AuthView.MODE.VERIFY_CODE + payload = + code: @code() + label: @client_repository.construct_cookie_label @phone_number_e164(), @client_type() + label_key: @client_repository.construct_cookie_label_key @phone_number_e164(), @client_type() + phone: @phone_number_e164() + return payload + + ############################################################################### + # Events + ############################################################################### + + changed_country: (view_model, event) => + @clear_error z.auth.AuthView.TYPE.PHONE + @country_code "+#{z.util.CountryCodes.get_country_code event?.currentTarget.value or @country()}" + $('#wire-login-phone').focus() + + changed_country_code: (view_model, event) => + country_code = (event?.currentTarget.value or @country_code()).match(/\d+/g)?.join('').substr 0, 4 + + if country_code + @country_code "+#{country_code}" + country_iso = z.util.CountryCodes.get_country_by_code(country_code) or 'X1' + else + @country_code '' + country_iso = 'X0' + + @country country_iso + $('#wire-login-phone').focus() + + changed_phone_number: => + input_value = @phone_number() + @phone_number (@phone_number().match(/\d+/g))?.join('') or '' + if @phone_number().length is 0 and input_value.length > 0 + @_add_error z.string.auth_error_phone_number_invalid, z.auth.AuthView.TYPE.PHONE + + clear_error: (mode, event) => @_remove_error event?.currentTarget.classList[1] or mode + + clear_error_password: (view_model, event) -> + return if event.keyCode is z.util.KEYCODE.ENTER + @failed_validation_password false + if event.currentTarget.value.length is 0 or event.currentTarget.value.length >= 8 + @_remove_error event.currentTarget.classList[1] + + clicked_on_change_email: => @_set_hash z.auth.AuthView.MODE.ACCOUNT_REGISTER + + clicked_on_change_phone: => @_set_hash z.auth.AuthView.MODE.ACCOUNT_PHONE + + clicked_on_login: => + @_set_hash z.auth.AuthView.MODE.ACCOUNT_LOGIN + $('#wire-login-phone').focus_field() if @visible_method() is z.auth.AuthView.MODE.ACCOUNT_PHONE + + clicked_on_login_email: => @_set_hash z.auth.AuthView.MODE.ACCOUNT_EMAIL + + clicked_on_login_phone: => @_set_hash z.auth.AuthView.MODE.ACCOUNT_PHONE + + clicked_on_password: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.PASSWORD_RESET, value: 'fromSignIn' + (window.open z.localization.Localizer.get_text z.string.url_password_reset)?.focus() + + clicked_on_register: => @_set_hash z.auth.AuthView.MODE.ACCOUNT_REGISTER + + clicked_on_resend_code: => + return if not @can_resend_code() + @sign_in_phone() + + clicked_on_resend_registration: => + return if not @can_resend_registration() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.RESENT_EMAIL_VERIFICATION + @_fade_in_icon_spinner() + + if not @pending_server_request() + @pending_server_request true + payload = @_create_payload z.auth.AuthView.MODE.POSTED_RESEND + @auth.repository.resend_activation payload + .then (response) => @_on_resend_success response + .catch (error) => @_on_resend_error error + + clicked_on_resend_verification: => + return if not @can_resend_verification + @_fade_in_icon_spinner() + + if not @pending_server_request() + @pending_server_request true + @user_service.change_own_email @email() + .then (response) => @_on_resend_success response + .catch => + @pending_server_request false + $('.icon-spinner').fadeOut() + setTimeout => + $('.icon-error').fadeIn() + @disabled_by_animation false + , TIMEOUT.SHORT + + clicked_on_retry_registration: => + return if not @can_register() + @_fade_in_icon_spinner() + + if not @pending_server_request() + @pending_server_request true + payload = @_create_payload z.auth.AuthView.MODE.ACCOUNT_REGISTER + @auth.repository.register payload + .then (response) => @_on_resend_success response + .catch (error) => @_on_resend_error error + + clicked_on_terms: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.NAVIGATION.OPENED_TERMS + (window.open z.localization.Localizer.get_text z.string.url_terms_of_use)?.focus() + + clicked_on_verify_later: => @_authentication_successful() + + clicked_on_wire_link: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.NAVIGATION.OPENED_WIRE_WEBSITE + (window.open z.localization.Localizer.get_text z.string.url_wire)?.focus() + + keydown_phone_code: (view_model, event) => + combo_key = if z.util.Environment.os.win then event.ctrlKey else event.metaKey + return true if combo_key and event.keyCode is z.util.KEYCODE.V + return false if event.altKey or event.ctrlKey or event.metaKey or event.shiftKey + + target_id = event.currentTarget.id + target_digit = window.parseInt(target_id.substr target_id.length - 1) + + switch event.keyCode + when z.util.KEYCODE.ARROW_LEFT, z.util.KEYCODE.ARROW_UP + focus_digit = target_digit - 1 + $("#wire-phone-code-digit-#{Math.max 1, focus_digit}").focus() + when z.util.KEYCODE.ARROW_DOWN, z.util.KEYCODE.ARROW_RIGHT + focus_digit = target_digit + 1 + $("#wire-phone-code-digit-#{Math.min 6, focus_digit}").focus() + when z.util.KEYCODE.BACKSPACE, z.util.KEYCODE.DELETE + if event.currentTarget.value is '' + focus_digit = target_digit - 1 + $("#wire-phone-code-digit-#{Math.max 1, focus_digit}").focus() + return true + else + char = String.fromCharCode(event.keyCode).match(/\d+/g) or String.fromCharCode(event.keyCode - 48).match /\d+/g + if char + @code_digits()[target_digit - 1] char + focus_digit = target_digit + 1 + $("#wire-phone-code-digit-#{Math.min 6, focus_digit}").focus() + + input_phone_code: (view_model, event) => + target_id = event.currentTarget.id + target_digit = window.parseInt(target_id.substr target_id.length - 1) + array_digit = target_digit - 1 + target_value = event.currentTarget.value + input_value = target_value.match(/\d+/g)?.join '' + + if input_value + focus_digit = target_digit + input_value.length + $("#wire-phone-code-digit-#{Math.min 6, focus_digit}").focus() + digits = input_value.substr(0, 6 - array_digit).split '' + target_value = digits[0] + @code_digits()[array_digit + i] digit for digit, i in digits + else + @code_digits()[array_digit] null + + clicked_on_manage_devices: => + @device_modal ?= new zeta.webapp.module.Modal '#modal-limit' + if @device_modal.is_hidden() + @client_repository.get_clients_for_self false + @device_modal.toggle() + + close_model_manage_devices: => @device_modal.toggle() + + click_on_remove_device_submit: (password, device) => + @client_repository.delete_client device.id, password + .then => + return @_register_client() + .then => + @device_modal.toggle() + .catch (error) => + @remove_form_error true + @logger.log @logger.levels.ERROR, "Unable to replace device: #{error?.message}", error + + click_on_history_confirm: => @_redirect_to_app() + + + ############################################################################### + # Callbacks + ############################################################################### + + _on_register_error: (error) => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.ENTERED_CREDENTIALS, + {outcome: 'fail'} + + @pending_server_request false + switch error.label + when z.service.BackendClientError::LABEL.BLACKLISTED_EMAIL, z.service.BackendClientError::LABEL.UNAUTHORIZED + @_add_error z.string.auth_error_email_forbidden, z.auth.AuthView.TYPE.EMAIL + when z.service.BackendClientError::LABEL.KEY_EXISTS + payload = @_create_payload z.auth.AuthView.MODE.ACCOUNT_EMAIL + @auth.repository.login payload, @persist() + .then => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.LOGGED_IN, + { + context: z.auth.AuthView.MODE.ACCOUNT_REGISTER + remember_me: @persist() + } + @_authentication_successful() + .catch => + @_add_error z.string.auth_error_email_exists, z.auth.AuthView.TYPE.EMAIL + @_has_errors() + return + when z.service.BackendClientError::LABEL.MISSING_IDENTITY + @_add_error z.string.auth_error_email_missing, z.auth.AuthView.TYPE.EMAIL + + if @_has_errors() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.ENTERED_CREDENTIALS, + {outcome: 'fail', reason: error.label} + return + + @get_wire false + if navigator.onLine + @_set_hash z.auth.AuthView.MODE.POSTED_RETRY + else + @_set_hash z.auth.AuthView.MODE.POSTED_OFFLINE + + _on_resend_error: (error) => + @pending_server_request false + $('.icon-spinner').fadeOut() + setTimeout => + $('.icon-error').fadeIn() + @_on_register_error error + @disabled_by_animation false + , TIMEOUT.SHORT + + _on_resend_success: => + @pending_server_request false + $('.icon-spinner').fadeOut() + setTimeout => + $('.icon-check').fadeIn() + @posted_mode z.auth.AuthView.MODE.POSTED_RESEND if @posted_mode() is z.auth.AuthView.MODE.POSTED_RETRY + , TIMEOUT.SHORT + setTimeout => + $('.icon-check').fadeOut() + $('.icon-envelope').fadeIn() + @disabled_by_animation false + , TIMEOUT.LONG + + _wait_for_activate: => + @logger.log 'Opened WebSocket connection to wait for account activation' + @web_socket_service.connect (notification) => + event = notification.payload[0] + @logger.log "»» Event: '#{event.type}'", {event_object: event, event_json: JSON.stringify event} + if event.type is z.event.Backend.USER.ACTIVATE + @_account_verified() + + _wait_for_update: => + @logger.log 'Opened WebSocket connection to wait for user update' + @web_socket_service.connect (notification) => + event = notification.payload[0] + @logger.log "»» Event: '#{event.type}'", {event_object: event, event_json: JSON.stringify event} + if event.type is z.event.Backend.USER.UPDATE and event.user.email + @_account_verified false + + + ############################################################################### + # Views and Navigation + ############################################################################### + + show_account_register: (focus = 'wire-register-name') -> + switch_params = + section: z.auth.AuthView.SECTION.ACCOUNT + mode: z.auth.AuthView.MODE.ACCOUNT_REGISTER + focus: focus + @switch_ui switch_params + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.OPENED_EMAIL_SIGN_UP, + {context: @registration_context} + + show_account_email: -> + switch_params = + section: z.auth.AuthView.SECTION.ACCOUNT + mode: z.auth.AuthView.MODE.ACCOUNT_LOGIN + method: z.auth.AuthView.MODE.ACCOUNT_EMAIL + focus: 'wire-login-email' + @switch_ui switch_params + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.OPENED_LOGIN, + {context: z.auth.AuthView.MODE.ACCOUNT_EMAIL} + + show_account_login: => + switch_params = + section: z.auth.AuthView.SECTION.ACCOUNT + mode: z.auth.AuthView.MODE.ACCOUNT_LOGIN + focus: 'wire-login-email' + @switch_ui switch_params + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.OPENED_LOGIN, + {context: @visible_method()} + + show_account_phone: -> + switch_params = + section: z.auth.AuthView.SECTION.ACCOUNT + mode: z.auth.AuthView.MODE.ACCOUNT_LOGIN + method: z.auth.AuthView.MODE.ACCOUNT_PHONE + focus: 'wire-login-phone' + @switch_ui switch_params + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ACCOUNT.OPENED_LOGIN, + {context: z.auth.AuthView.MODE.ACCOUNT_PHONE} + + show_verify_code: => + if not z.util.is_valid_phone_number @phone_number_e164() + return @_set_hash z.auth.AuthView.MODE.ACCOUNT_PHONE + switch_params = + section: z.auth.AuthView.SECTION.VERIFY + mode: z.auth.AuthView.MODE.VERIFY_CODE + focus: 'wire-phone-code-digit-1' + @switch_ui switch_params + $('#wire-phone-code-digit-1').focus() + + show_verify_mail: -> + switch_params = + section: z.auth.AuthView.SECTION.VERIFY + mode: z.auth.AuthView.TYPE.EMAIL + focus: 'wire-mail-email' + @switch_ui switch_params + + show_posted_offline: -> + @_show_icon_error() + switch_params = + section: z.auth.AuthView.SECTION.POSTED + mode: z.auth.AuthView.MODE.POSTED_OFFLINE + @switch_ui switch_params + + show_posted_pending: -> + @_show_icon_envelope() + switch_params = + section: z.auth.AuthView.SECTION.POSTED + mode: z.auth.AuthView.MODE.POSTED_PENDING + @switch_ui switch_params + + show_posted_resend: -> + @_show_icon_envelope() + switch_params = + section: z.auth.AuthView.SECTION.POSTED + mode: z.auth.AuthView.MODE.POSTED_RESEND + @switch_ui switch_params + + show_posted_retry: -> + @_show_icon_error() + switch_params = + section: z.auth.AuthView.SECTION.POSTED + mode: z.auth.AuthView.MODE.POSTED_RETRY + @switch_ui switch_params + + show_posted_verify: -> + @_show_icon_envelope() + switch_params = + section: z.auth.AuthView.SECTION.POSTED + mode: z.auth.AuthView.MODE.POSTED_VERIFY + @switch_ui switch_params + + show_limit: -> + switch_params = + section: z.auth.AuthView.SECTION.LIMIT + mode: z.auth.AuthView.MODE.LIMIT + @switch_ui switch_params + + show_history: -> + switch_params = + section: z.auth.AuthView.SECTION.HISTORY + mode: z.auth.AuthView.MODE.HISTORY + @switch_ui switch_params + + + ############################################################################### + # Animations + ############################################################################### + + switch_ui: (switch_params) => + if @show_initial_animation + direction = z.auth.AuthView.ANIMATION_DIRECTION.VERTICAL_TOP + else if @visible_section() is z.auth.AuthView.SECTION.ACCOUNT + if switch_params.section isnt z.auth.AuthView.SECTION.ACCOUNT + direction = z.auth.AuthView.ANIMATION_DIRECTION.HORIZONTAL_LEFT + else if @visible_section() is z.auth.AuthView.SECTION.POSTED + if switch_params.section is z.auth.AuthView.SECTION.ACCOUNT + direction = z.auth.AuthView.ANIMATION_DIRECTION.HORIZONTAL_RIGHT + else if @visible_section() is z.auth.AuthView.SECTION.VERIFY + if switch_params.section is z.auth.AuthView.SECTION.ACCOUNT + direction = z.auth.AuthView.ANIMATION_DIRECTION.HORIZONTAL_RIGHT + else if switch_params.section is z.auth.AuthView.SECTION.POSTED + direction = z.auth.AuthView.ANIMATION_DIRECTION.HORIZONTAL_LEFT + else if @visible_mode() is z.auth.AuthView.MODE.VERIFY_CODE + if switch_params.mode is z.auth.AuthView.TYPE.EMAIL + direction = z.auth.AuthView.ANIMATION_DIRECTION.HORIZONTAL_LEFT + + if switch_params.section is z.auth.AuthView.SECTION.ACCOUNT + @account_mode switch_params.mode + else if switch_params.section is z.auth.AuthView.SECTION.POSTED + @posted_mode switch_params.mode + + @_clear_animations z.auth.AuthView.TYPE.SECTION + if switch_params.section isnt @visible_section() + animation_params = + type: z.auth.AuthView.TYPE.SECTION + section: switch_params.section + direction: direction + @_shift_ui animation_params + + @_clear_animations z.auth.AuthView.TYPE.FORM + if switch_params.mode isnt @visible_mode() + animation_params = + type: z.auth.AuthView.TYPE.FORM + section: switch_params.section + selector: switch_params.mode + direction: direction + @_shift_ui animation_params + + if not switch_params.method and not @visible_method() + @_show_method z.auth.AuthView.MODE.ACCOUNT_EMAIL + @visible_method z.auth.AuthView.MODE.ACCOUNT_EMAIL + else if switch_params.method and @visible_method() isnt switch_params.method + @_show_method switch_params.method + @visible_method switch_params.method + + $("##{switch_params.focus}").focus_field() if switch_params.focus + + _show_method: (method) -> + @_clear_errors() + $('.selector-method').find('.button').removeClass 'is-active' + $(".btn-login-#{method}").addClass 'is-active' + $('.method:visible').hide() + .css opacity: 0 + $("#login-method-#{method}").show() + .css opacity: 1 + + _shift_ui: (animation_params) => + direction = animation_params.direction + old_component = $(".#{animation_params.type}:visible") + new_component = $("##{animation_params.type}-#{animation_params.section}") + if animation_params.selector + new_component = $("##{animation_params.type}-#{animation_params.section}-#{animation_params.selector}") + new_component.show() + + _change_visible = => + switch animation_params.type + when z.auth.AuthView.TYPE.FORM then @visible_mode animation_params.selector + when z.auth.AuthView.TYPE.SECTION then @visible_section animation_params.section + + if not animation_params.direction + old_component.css + display: '' + opacity: '' + new_component.css opacity: 1 + _change_visible() + else if old_component.length is 0 + @disabled_by_animation true + + requestAnimFrame => + new_anim = $.Deferred() + + new_component + .addClass "incoming-#{animation_params.direction}" + .one z.util.alias.animationend, -> + new_anim.resolve() + $(@).css opacity: 1 + + $.when(new_anim).then => + _change_visible() + @disabled_by_animation false + @show_initial_animation = false + else + @disabled_by_animation true + + requestAnimFrame => + old_anim = $.Deferred() + new_anim = $.Deferred() + + $(old_component[0]) + .addClass "outgoing-#{animation_params.direction}" + .one z.util.alias.animationend, -> + old_anim.resolve() + $(@).css + display: '' + opacity: '' + new_component + .addClass "incoming-#{animation_params.direction}" + .one z.util.alias.animationend, -> + new_anim.resolve() + $(@).css opacity: 1 + + $.when(old_anim, new_anim).then => + _change_visible() + @disabled_by_animation false + + _clear_animations: (type = z.auth.AuthView.TYPE.FORM) -> + $(".#{type}") + .off z.util.alias.animationend + .removeClass (index, css) -> (css.match(/\boutgoing-\S+/g) or []).join ' ' + .removeClass (index, css) -> (css.match(/\bincoming-\S+/g) or []).join ' ' + + _fade_in_icon_spinner: => + @disabled_by_animation true + $('.icon-envelope').fadeOut() + $('.icon-error').fadeOut() + $('.icon-spinner').fadeIn() + + _show_icon_envelope: -> + $('.icon-error').hide() + $('.icon-envelope').show() + + _show_icon_error: -> + $('.icon-envelope').hide() + $('.icon-error').show() + + + ############################################################################### + # URL changes + ############################################################################### + + ### + Set location hash + @private + @param hash [String] Hash value + ### + _set_hash: (hash = '') -> window.location.hash = hash + + ### + Get location hash + @private + @return [String] Hash value + ### + _get_hash: -> return window.location.hash.substr 1 + + ### + No hash value + @private + @return [Boolean] No location hash value + ### + _has_no_hash: -> return window.location.hash.length is 0 + + ### + Navigation on hash change + @private + ### + _on_hash_change: => + @_clear_errors() + switch @_get_hash() + when z.auth.AuthView.MODE.ACCOUNT_EMAIL then @show_account_email() + when z.auth.AuthView.MODE.ACCOUNT_LOGIN then @show_account_login() + when z.auth.AuthView.MODE.VERIFY_CODE then @show_verify_code() + when z.auth.AuthView.MODE.VERIFY_ADD_EMAIL then @show_verify_mail() + when z.auth.AuthView.MODE.POSTED then @show_posted_resend() + when z.auth.AuthView.MODE.POSTED_OFFLINE then @show_posted_offline() + when z.auth.AuthView.MODE.POSTED_PENDING then @show_posted_pending() + when z.auth.AuthView.MODE.POSTED_RETRY then @show_posted_retry() + when z.auth.AuthView.MODE.POSTED_VERIFY then @show_posted_verify() + when z.auth.AuthView.MODE.LIMIT then @show_limit() + when z.auth.AuthView.MODE.HISTORY then @show_history() + else @show_account_register() + + + ############################################################################### + # Validation errors + ############################################################################### + + ### + Add a validation error. + + @private + @param string_identifier [String] Identifier of error message + @param types [Array | String] Input type(s) of validation error + ### + _add_error: (string_identifier, types) -> + error = new z.auth.ValidationError types or [], string_identifier + @validation_errors.push error + for type in error.types + switch type + when z.auth.AuthView.TYPE.CODE then @failed_validation_code true + when z.auth.AuthView.TYPE.EMAIL then @failed_validation_email true + when z.auth.AuthView.TYPE.NAME then @failed_validation_name true + when z.auth.AuthView.TYPE.PASSWORD then @failed_validation_password true + when z.auth.AuthView.TYPE.PHONE then @failed_validation_phone true + when z.auth.AuthView.TYPE.TERMS then @failed_validation_terms true + + ### + Removes all validation errors. + @private + ### + _clear_errors: -> + @failed_validation_code false + @failed_validation_email false + @failed_validation_name false + @failed_validation_password false + @failed_validation_phone false + @failed_validation_terms false + @validation_errors [] + + ### + Get the validation error by inout type. + @param type [z.auth.AuthView.TYPE] Input type to get error for + @return [z.auth.ValidationError] Validation Error + ### + _get_error_by_type: (type) => + return ko.utils.arrayFirst @validation_errors(), (error) -> + type in error.types + + ### + Check whether a form has errors and play the alert sound. + @private + @return [Boolean] Does the form have an error + ### + _has_errors: -> + has_error = false + if @validation_errors().length > 0 + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + has_error = true + return has_error + + ### + Remove a validation error. + @private + @param type [String] Input type of validation error + ### + _remove_error: (type) -> + @validation_errors.remove @_get_error_by_type type + switch type + when z.auth.AuthView.TYPE.CODE then @failed_validation_code false + when z.auth.AuthView.TYPE.EMAIL then @failed_validation_email false + when z.auth.AuthView.TYPE.NAME then @failed_validation_name false + when z.auth.AuthView.TYPE.PASSWORD then @failed_validation_password false + when z.auth.AuthView.TYPE.PHONE then @failed_validation_phone false + when z.auth.AuthView.TYPE.TERMS then @failed_validation_terms false + + ### + Validate code input. + @private + @return [Boolean] Phone code is long enough + ### + _validate_code: -> + return @code().length >= 6 + + ### + Validate email input. + @private + ### + _validate_email: -> + if @email().length is 0 + @_add_error z.string.auth_error_email_missing, z.auth.AuthView.TYPE.EMAIL + else if not z.util.is_valid_email @email() + @_add_error z.string.auth_error_email_malformed, z.auth.AuthView.TYPE.EMAIL + + ### + Validate the user input. + + @private + @param mode [z.auth.AuthView.MODE] View state of the authentication page + @return [Boolean] Does the user input have validation errors + ### + _validate_input: (mode) -> + @_clear_errors() + + if mode is z.auth.AuthView.MODE.ACCOUNT_REGISTER + @_validate_name() + + email_and_password_modes = [ + z.auth.AuthView.MODE.ACCOUNT_EMAIL + z.auth.AuthView.MODE.ACCOUNT_REGISTER + z.auth.AuthView.MODE.VERIFY_ADD_EMAIL + ] + if mode in email_and_password_modes + @_validate_email() + @_validate_password mode + + if mode is z.auth.AuthView.MODE.ACCOUNT_PHONE + @_validate_phone() + + if mode is z.auth.AuthView.MODE.ACCOUNT_REGISTER + @_validate_terms_of_use() + + return not @_has_errors() + + ### + Validate name input. + @private + ### + _validate_name: -> + if @name().length < z.config.MINIMUM_USERNAME_LENGTH + @_add_error z.string.auth_error_name_short, z.auth.AuthView.TYPE.NAME + + ### + Validate password input. + @private + @param mode [z.auth.AuthView.MODE] View state of the authentication page + ### + _validate_password: (mode) -> + if @password().length < z.config.MINIMUM_PASSWORD_LENGTH + if mode is z.auth.AuthView.MODE.ACCOUNT_EMAIL + return @_add_error z.string.auth_error_password_wrong, z.auth.AuthView.TYPE.PASSWORD + @_add_error z.string.auth_error_password_short, z.auth.AuthView.TYPE.PASSWORD + + ### + Validate phone input. + @private + ### + _validate_phone: -> + if not z.util.is_valid_phone_number(@phone_number_e164()) and z.util.Environment.backend.current is 'production' + @_add_error z.string.auth_error_phone_number_invalid, z.auth.AuthView.TYPE.PHONE + + ### + Validate terms of use. + @private + ### + _validate_terms_of_use: -> + if not @accepted_terms_of_use() + @_add_error z.string.auth_error_terms_of_use, z.auth.AuthView.TYPE.TERMS + + + ############################################################################### + # Misc + ############################################################################### + + ### + Logout the user again. + @todo: What do we actually need to delete here + ### + logout: => + @auth.repository.logout() + .then => + @auth.repository.delete_access_token() + window.location.replace '/auth' + + ### + User account has been verified. + @private + @param registration [Boolean] Verification from registration + ### + _account_verified: (registration = true) => + @logger.log @logger.levels.INFO, 'User account verified. User can now login.' + if registration + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.REGISTRATION.SUCCEEDED, + {content: @registration_context} + @_authentication_successful() + + ### + User successfully authenticated on the backend side + @note Gets the client and forwards the user to the login. + @private + ### + _authentication_successful: => + @logger.log @logger.levels.INFO, 'Logging in' + @_get_self_user() + .then => + return @client_repository.get_valid_local_client() + .catch (error) => + @logger.log @logger.levels.INFO, "No valid local client found: #{error.message}", error + if error.type is z.client.ClientError::TYPE.MISSING_ON_BACKEND + @logger.log @logger.levels.INFO, 'Local client rejected as invalid by backend. Reinitializing storage.' + @storage_service.init @self_user().id + .then => + return @storage_repository.init true + .then => + return @cryptography_repository.init() + .then => + if @client_repository.current_client() + @logger.log @logger.levels.INFO, 'Active client found. Redirecting to app...' + @_redirect_to_app() + else + @logger.log @logger.levels.INFO, 'No active client found. We need to register one...' + @_register_client() + .catch (error) => + @logger.log @logger.levels.ERROR, "Login failed: #{error?.message}", error + @_add_error z.string.auth_error_misc + @_has_errors() + @_set_hash z.auth.AuthView.MODE.ACCOUNT_LOGIN + + ### + Get and store the self user. + @private + @return [Promise] Self user + ### + _get_self_user: -> + return new Promise (resolve, reject) => + @user_repository.get_me() + .then (user_et) => + @self_user user_et + @logger.log @logger.levels.INFO, "Got self user: #{@self_user().id}" + @pending_server_request false + + if @self_user().email()? + @storage_service.init @self_user().id + .then => + @client_repository.init @self_user() + resolve @self_user() + .catch (error) -> reject error + else + @_set_hash z.auth.AuthView.MODE.VERIFY_ADD_EMAIL + + ### + Redirects to the app after successful login + @private + ### + _redirect_to_app: => + url = '/' + url = "/#{@auth.settings.parameter}" if @auth.settings.parameter? + connect_token = z.util.get_url_parameter z.auth.URLParameter.CONNECT + url = z.util.append_url_parameter url, "#{z.auth.URLParameter.CONNECT}=#{connect_token}" if connect_token + window.location.replace url + + _register_client: => + @client_repository.register_client @password() + .then (client_observable) => + @event_repository.current_client = client_observable + @event_repository.get_last_notification_id() + .then (last_notification_id) => + @event_repository.last_notification_id last_notification_id + @logger.log @logger.levels.INFO, "Set starting point on notification stream to '#{last_notification_id}'" + .catch (error) => + if error.code is z.service.BackendClientError::STATUS_CODE.NOT_FOUND + @logger.log @logger.levels.WARN, + "Cannot set starting point on notification stream: #{error.message}", error + else + throw error + .then => + return @client_repository.get_clients_for_self() + .then (client_ets) => + @logger.log @logger.levels.INFO, "User has '#{client_ets?.length}' registered clients", client_ets + + # Show history screen if there are already registered clients + if client_ets?.length > 0 + @_set_hash z.auth.AuthView.MODE.HISTORY + # Make sure client entities always see the history screen + else if @client_repository.current_client().is_temporary() + @_set_hash z.auth.AuthView.MODE.HISTORY + # Don't show history screen if the webapp is the first client that has been registered + else + @_redirect_to_app() + .catch (error) => + if error.type is z.client.ClientError::TYPE.TOO_MANY_CLIENTS + @logger.log @logger.levels.WARN, 'User has already registered the maximum number of clients', error + window.location.hash = z.auth.AuthView.MODE.LIMIT + else + @logger.log @logger.levels.ERROR, "Failed to register a new client: #{error.message}", error + + ### + Track app launch for Localytics + @private + ### + _track_app_launch: -> + mechanism = 'direct' + if document.referrer.startsWith 'https://wire.com/verify/' + mechanism = 'email_verify' + else if document.referrer.startsWith 'https://wire.com/forgot/' + mechanism = 'password_reset' + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.APP_LAUNCH, mechanism: mechanism + +$ -> + if $('.auth-page').length isnt 0 + wire.auth.view = new z.ViewModel.AuthViewModel 'auth-page', wire.auth + +# jQuery helpers +$.fn.extend + focus_field: -> + @each -> + # Timeout needed (for Chrome): http://stackoverflow.com/a/17384592/451634 + setTimeout => + $(@).focus() + , 0 + + # FIX to prevent unwanted auto form fill on Chrome + prevent_prefill: -> + if z.util.Environment.browser.chrome or z.util.Environment.browser.opera + @each -> + $(@) + .attr 'readonly', true + .on 'focus', -> + $(@).removeAttr 'readonly' diff --git a/app/script/view_model/BackgroundViewModel.coffee b/app/script/view_model/BackgroundViewModel.coffee new file mode 100644 index 00000000000..c64332d185c --- /dev/null +++ b/app/script/view_model/BackgroundViewModel.coffee @@ -0,0 +1,63 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +class z.ViewModel.BackgroundViewModel + constructor: (element_id, @content, @conversation_repository, @user_repository) -> + + @webapp_loaded = ko.observable false + + @self_user = ko.computed => + @user_repository.self()?.picture_medium_url() if @webapp_loaded() + + @background_list_element = $("##{element_id}") + @background_list_element.on z.util.alias.animationend, -> + if not $(@).hasClass 'background-fullscreen' + $('.conversation, .connect-requests-wrapper').css 'z-index': '' + + @content.state.subscribe => + if @content.state() is z.ViewModel.CONTENT_STATE.PROFILE + requestAnimFrame => + @background_list_element.addClass 'background-fullscreen' + requestAnimFrame => + if @background_list_element.hasClass 'background-fullscreen-anim-disabled' + @background_list_element.removeClass 'background-fullscreen-anim-disabled' + $('.conversation, .connect-requests-wrapper').css 'z-index': '' + else + if @background_list_element.hasClass 'background-fullscreen' + @background_list_element.removeClass 'background-fullscreen' + $('.conversation, .connect-requests-wrapper').css 'z-index': '-1' + + @blur_background = _.throttle (ratio) => + blur_radius = (ratio * 24) | 0 + blur_css = "blur(#{blur_radius}px)" + requestAnimFrame => + @background_list_element.find('.background').css + '-webkit-filter': blur_css + 'filter': blur_css + , 50 + + amplify.subscribe z.event.WebApp.LOADED, => @webapp_loaded true + amplify.subscribe z.event.WebApp.LIST.BLUR, @blur_background + amplify.subscribe z.event.WebApp.LIST.FULLSCREEN_ANIM_DISABLED, => + @background_list_element.addClass 'background-fullscreen-anim-disabled' + + ko.applyBindings @, document.getElementById element_id diff --git a/app/script/view_model/CallShortcutsViewModel.coffee b/app/script/view_model/CallShortcutsViewModel.coffee new file mode 100644 index 00000000000..483e074490f --- /dev/null +++ b/app/script/view_model/CallShortcutsViewModel.coffee @@ -0,0 +1,70 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +### +Last remainder of the CallBannerViewModel. +@todo Move functionality elsewhere and remove +### +class z.ViewModel.CallShortcutsViewModel + constructor: (@call_center) -> + @logger = new z.util.Logger 'z.ViewModel.CallShortcutsViewModel', z.config.LOGGER.OPTIONS + + @joined_call = @call_center.joined_call + + @joined_call.subscribe (call_et) => + @_update_shortcut_subscription call_et + + ########################### + # Shortcuts + ########################### + + _update_shortcut_subscription: (call_et) => + @_unsubscribe_shortcuts() + return if not call_et + + switch call_et.state() + when z.calling.enum.CallState.ONGOING, z.calling.enum.CallState.OUTGOING + @_subscribe_shortcuts_outgoing_ongoing() + when z.calling.enum.CallState.INCOMING + @_subscribe_shortcuts_incoming() + + conversation_name = call_et.conversation_et.display_name() + @logger.log @logger.levels.DEBUG, "Updated call shortcuts for '#{call_et.state()}' call in conversation '#{call_et.id}' (#{conversation_name})" + + _subscribe_shortcuts_incoming: => + amplify.subscribe z.event.WebApp.SHORTCUT.CALL_IGNORE, @on_ignore_call + + _subscribe_shortcuts_outgoing_ongoing: => + amplify.subscribe z.event.WebApp.SHORTCUT.CALL_MUTE, @on_mute_call + + _unsubscribe_shortcuts: => + amplify.unsubscribe z.event.WebApp.SHORTCUT.CALL_MUTE, @on_mute_call + amplify.unsubscribe z.event.WebApp.SHORTCUT.CALL_IGNORE, @on_ignore_call + + ########################### + # Component actions + ########################### + + on_ignore_call: => + @call_center.state_handler.ignore_call @joined_call()?.id + + on_mute_call: => + @call_center.state_handler.toggle_audio @joined_call()?.id diff --git a/app/script/view_model/ConnectRequestsViewModel.coffee b/app/script/view_model/ConnectRequestsViewModel.coffee new file mode 100644 index 00000000000..e860402ec38 --- /dev/null +++ b/app/script/view_model/ConnectRequestsViewModel.coffee @@ -0,0 +1,52 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +# View model for connection requests. +class z.ViewModel.ConnectRequestsViewModel + constructor: (element_id, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ConnectRequestsViewModel', z.config.LOGGER.OPTIONS + + @connect_requests = @user_repository.connect_requests + + ### + Click on accept. + @param user_et [z.entity.User] User to accept connection request from + @return [Promise] Promise that resolves when the connection request was accepted + ### + click_on_accept: (user_et) => + @user_repository.accept_connection_request user_et, @connect_requests().length is 1 + + ### + Click on ignore. + @param user_et [z.entity.User] User to ignore connection request from + @return [Promise] Promise that resolves when the connection request was ignored + ### + click_on_ignore: (user_et) => + @user_repository.ignore_connection_request user_et + + ### + Called after each connection request is rendered. + @param elements [Object] rendered objects + @param request [z.entity.User] Rendered connection request + ### + after_render: (elements, request) => + if z.util.array_is_last @connect_requests(), request + requestAnimFrame -> $('.connect-requests').scroll_to_bottom() diff --git a/app/script/view_model/ConversationInputViewModel.coffee b/app/script/view_model/ConversationInputViewModel.coffee new file mode 100644 index 00000000000..749fec74f68 --- /dev/null +++ b/app/script/view_model/ConversationInputViewModel.coffee @@ -0,0 +1,170 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +# Parent: z.ViewModel.RightViewModel +class z.ViewModel.ConversationInputViewModel + constructor: (element_id, @conversation_repository, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ConversationInputViewModel', z.config.LOGGER.OPTIONS + + @conversation_et = @conversation_repository.active_conversation + @conversation_et.subscribe => @conversation_has_focus true + + @self = @user_repository.self + @list_not_bottom = ko.observable true + + @conversation_has_focus = ko.observable(true).extend notify: 'always' + @browser_has_focus = ko.observable true + + @blinking_cursor = ko.computed => + return @browser_has_focus() and @conversation_has_focus() + + @has_text_input = ko.computed => + return @conversation_et()?.input().length > 0 + + @show_giphy_button = ko.computed => + return @has_text_input() and @conversation_et()?.input().length <= 15 + + @ping_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_ping + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.PING} + } + @picture_tooltip = z.localization.Localizer.get_text z.string.tooltip_conversation_picture + @file_tooltip = z.localization.Localizer.get_text z.string.tooltip_conversation_file + + @ping_disabled = ko.observable false + + $(window) + .blur => @browser_has_focus false + .focus => @browser_has_focus true + + @_init_subscriptions() + + _init_subscriptions: -> + amplify.subscribe z.event.WebApp.SEARCH.SHOW, => @conversation_has_focus false + amplify.subscribe z.event.WebApp.SEARCH.HIDE, => window.requestAnimFrame => @conversation_has_focus true + amplify.subscribe z.event.WebApp.EXTENSIONS.GIPHY.SEND, => @conversation_et()?.input '' + amplify.subscribe z.event.WebApp.CONVERSATION.IMAGE.SEND, @upload_images + + added_to_view: => + setTimeout => + amplify.subscribe z.event.WebApp.SHORTCUT.PING, => @ping() + , 50 + + removed_from_view: -> + amplify.unsubscribe z.event.WebApp.SHORTCUT.PING + + ping: => + return if @ping_disabled() + + @ping_disabled true + @conversation_repository.send_encrypted_knock @conversation_et() + .then => + window.setTimeout => + @ping_disabled false + , 2000 + + toggle_extensions_menu: -> + amplify.publish z.event.WebApp.EXTENSIONS.GIPHY.SHOW + + send_message: (data, event) => + message = z.util.trim_line_breaks @conversation_et().input() + if message.length is 0 + return + + if message.length > z.config.MAXIMUM_MESSAGE_LENGTH + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.TOO_LONG_MESSAGE, + data: z.config.MAXIMUM_MESSAGE_LENGTH + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.CHARACTER_LIMIT_REACHED, + characters: message.length + return + + link_data = z.links.LinkPreviewHelpers.get_first_link_with_offset message + if link_data? + [url, offset] = link_data + @conversation_repository.send_encrypted_message_with_link_preview message, url, offset, @conversation_et() + else + @conversation_repository.send_encrypted_message message, @conversation_et() + + @conversation_et().input '' + $(event.target).focus() + + upload_images: (images) => + for image in images + return @_show_upload_warning image if image.size > z.config.MAXIMUM_IMAGE_FILE_SIZE + + @conversation_repository.upload_images @conversation_et(), images + + _show_upload_warning: (image) -> + warning = z.localization.Localizer.get_text { + id: if image.type is 'image/gif' then z.string.alert_gif_too_large else z.string.alert_upload_too_large + replace: {placeholder: '%no', content: z.config.MAXIMUM_IMAGE_FILE_SIZE / 1024 / 1024} + } + + attributes = + reason: 'too large' + type: image.type + size: image.size + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.IMAGE_SENT_ERROR, attributes + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + setTimeout -> + window.alert warning + , 200 + + upload_files: (files) => + pending_uploads = @conversation_repository.get_number_of_pending_uploads() + if pending_uploads + files.length > z.config.MAXIMUM_ASSET_UPLOADS + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.UPLOAD_PARALLEL, + data: z.config.MAXIMUM_ASSET_UPLOADS + return + + for file in files + if file.size > z.config.MAXIMUM_ASSET_FILE_SIZE + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.FILE.UPLOAD_TOO_BIG, + {size: file.size, type: file.type} + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + setTimeout -> + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.UPLOAD_TOO_LARGE, + data: z.util.format_bytes(z.config.MAXIMUM_ASSET_FILE_SIZE) + , 200 + return + + @conversation_repository.upload_files @conversation_et(), files + + scroll_message_list: (list_height_new, list_height_old) -> + diff = list_height_new - list_height_old + input_height = $('.conversation-input').height() + is_scrolled_bottom = $('.messages-wrap').is_scrolled_bottom() + + $('.message-list') + .css 'bottom', input_height + .data('antiscroll')?.rebuild() + + if is_scrolled_bottom + $('.messages-wrap').scroll_to_bottom() + else + $('.messages-wrap').scroll_by diff + + show_separator: (is_scrolled_bottom) => + @list_not_bottom not is_scrolled_bottom + + on_input_click: => + if not @has_text_input() + $('.messages-wrap').scroll_to_bottom() diff --git a/app/script/view_model/ConversationListViewModel.coffee b/app/script/view_model/ConversationListViewModel.coffee new file mode 100644 index 00000000000..a4a43c0673c --- /dev/null +++ b/app/script/view_model/ConversationListViewModel.coffee @@ -0,0 +1,202 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.ConversationListViewModel + ### + @param element_id [String] HTML selector + @param content [z.ViewModel.RightViewModel] View model + @param call_center [z.calling.CallCenter] Call center + @param user_repository [z.user.UserRepository] User repository + @param conversation_repository [z.conversation.ConversationRepository] Conversation repository + ### + constructor: (element_id, @content, @call_center, @user_repository, @conversation_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ConversationListViewModel', z.config.LOGGER.OPTIONS + + @selected_list_item = @content.state + @selected_conversation = ko.observable() + @status = + call: ko.computed => + call_et = @call_center.joined_call() + call_status = 'none' + + if call_et?.self_user_joined() + call_status = 'participating-in-group-call' + else + call_status = 'not-participating-in-group-call' + + return call_status + + @user = @user_repository.self + @show_badge = ko.observable false + + @connect_requests = @user_repository.connect_requests + @connect_requests_text = ko.computed => + count = @connect_requests().length + if count > 1 + return z.localization.Localizer.get_text { + id: z.string.conversation_list_many_connection_request + replace: {placeholder: '%no', content: count} + } + else + return z.localization.Localizer.get_text z.string.conversation_list_one_connection_request + + @is_conversation_list_visible = ko.observable true + @is_self_profile_visible = ko.observable false + + @conversations_calls = @conversation_repository.conversations_call + @conversations_archived = @conversation_repository.conversations_archived + @conversations_unarchived = @conversation_repository.conversations_unarchived + + @joined_call = @call_center.joined_call + + @webapp_is_loaded = ko.observable false + + @should_update_scrollbar = (ko.computed => + return @webapp_is_loaded() or + @is_conversation_list_visible() or + @conversations_unarchived().length or + @connect_requests().length or + @conversations_calls().length + ).extend notify: 'always', rateLimit: 500 + + @active_conversation_id = ko.computed => + if @conversation_repository.active_conversation()? + @conversation_repository.active_conversation().id + + @archive_tooltip = ko.computed => + return z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_list_archived + replace: {placeholder: '%no', content: @conversations_archived().length} + } + + @start_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_list_tooltip_start + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.START} + } + + @self_stream_state = @call_center.media_stream_handler.self_stream_state + + @show_toggle_screen = ko.pureComputed -> + return z.calling.CallCenter.supports_screen_sharing() + @disable_toggle_screen = ko.pureComputed => + return @joined_call()?.is_remote_screen_shared() + + @_init_subscriptions() + + ko.applyBindings @, document.getElementById element_id + + click_on_actions: (conversation_et, event) -> + amplify.publish z.event.WebApp.ACTION.SHOW, conversation_et, event + + click_on_connect_requests: -> + return if @selected_list_item() is z.ViewModel.CONTENT_STATE.PENDING + @is_self_profile_visible false + amplify.publish z.event.WebApp.PENDING.SHOW + + click_on_conversation: (conversation_et) => + return if @_is_selected_conversation conversation_et + @is_self_profile_visible false + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + + _init_subscriptions: => + amplify.subscribe z.event.WebApp.SEARCH.BADGE.SHOW, => @show_badge true + amplify.subscribe z.event.WebApp.SEARCH.BADGE.HIDE, => @show_badge false + amplify.subscribe z.event.WebApp.SHORTCUT.NEXT, @_go_to_next_conversation + amplify.subscribe z.event.WebApp.SHORTCUT.PREV, @_go_to_prev_conversation + amplify.subscribe z.event.WebApp.SHORTCUT.START, @click_on_people_button + amplify.subscribe z.event.WebApp.LOADED, @on_webapp_loaded + amplify.subscribe z.event.WebApp.CONVERSATION_LIST.SHOW, @_show + amplify.subscribe z.event.WebApp.ARCHIVE.CLOSE, @_show + amplify.subscribe z.event.WebApp.ARCHIVE.SHOW, @_hide + amplify.subscribe z.event.WebApp.SEARCH.SHOW, @_hide + + _go_to_next_conversation: => + conversations = @conversation_repository.conversations_unarchived() + index = conversations.indexOf(@conversation_repository.active_conversation()) - 1 + next_conversation_et = conversations[index] + amplify.publish z.event.WebApp.CONVERSATION.SHOW, next_conversation_et if next_conversation_et + + _go_to_prev_conversation: => + conversations = @conversation_repository.conversations_unarchived() + index = conversations.indexOf(@conversation_repository.active_conversation()) + 1 + prev_conversation_et = conversations[index] + amplify.publish z.event.WebApp.CONVERSATION.SHOW, prev_conversation_et if prev_conversation_et + + _is_selected_conversation: (conversation_et) => + @selected_list_item() is z.ViewModel.CONTENT_STATE.CONVERSATION and conversation_et.id is @active_conversation_id() + + on_webapp_loaded: => + @webapp_is_loaded true + + +############################################################################### +# Call stuff +############################################################################### + + on_accept_call: (conversation_et) => + @call_center.state_handler.join_call conversation_et.id, false + + on_accept_video: (conversation_et) => + @call_center.state_handler.join_call conversation_et.id, true + + on_cancel_call: (conversation_et) => + @call_center.state_handler.leave_call conversation_et.id + + on_ignore_call: (conversation_et) => + @call_center.state_handler.ignore_call conversation_et.id + + on_toggle_audio: (conversation_et) => + @call_center.state_handler.toggle_audio conversation_et.id + + on_toggle_screen: (conversation_et) -> + amplify.publish z.event.WebApp.CALL.STATE.TOGGLE_SCREEN, conversation_et.id + + on_toggle_video: (conversation_et) => + @call_center.state_handler.toggle_video conversation_et.id + +############################################################################### +# Footer actions +############################################################################### + click_on_archived_button: -> + amplify.publish z.event.WebApp.ARCHIVE.SHOW + + click_on_settings_button: => + if @is_self_profile_visible() + amplify.publish z.event.WebApp.PROFILE.HIDE + @is_self_profile_visible false + else + amplify.publish z.event.WebApp.PROFILE.SHOW + @is_self_profile_visible true + + click_on_people_button: -> + amplify.publish z.event.WebApp.SEARCH.SHOW + +############################################################################### +# Conversation List animations +############################################################################### + _hide: -> + $('.conversation-list').addClass 'conversation-list-is-hidden' + + _show: => + $('#conversation-list').removeClass('conversation-list-is-hidden') + + @is_conversation_list_visible true + @is_self_profile_visible false diff --git a/app/script/view_model/ConversationTitlebarViewModel.coffee b/app/script/view_model/ConversationTitlebarViewModel.coffee new file mode 100644 index 00000000000..c511424b0dd --- /dev/null +++ b/app/script/view_model/ConversationTitlebarViewModel.coffee @@ -0,0 +1,69 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +# Parent: z.ViewModel.ConversationTitlebarViewModel +class z.ViewModel.ConversationTitlebarViewModel + constructor: (element_id, @conversation_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ConversationTitlebarViewModel', z.config.LOGGER.OPTIONS + + # TODO remove this for now to ensure that buttons are clickable in osx wrappers + window.setTimeout -> + $('.titlebar').remove() + , 1000 + + @conversation_et = @conversation_repository.active_conversation + + @show_call_controls = ko.computed => + return false if not @conversation_et() + is_supported_conversation = @conversation_et().is_group() or @conversation_et().is_one2one() + is_active_conversation = @conversation_et().participating_user_ids().length and not @conversation_et().removed_from_conversation() + is_self_client_joined = @conversation_et().call()?.self_client_joined() + return is_supported_conversation and is_active_conversation and not is_self_client_joined + + @people_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_conversation_people + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.PEOPLE} + } + + added_to_view: => + setTimeout => + amplify.subscribe z.event.WebApp.SHORTCUT.PEOPLE, => @show_participants() + amplify.subscribe z.event.WebApp.SHORTCUT.ADD_PEOPLE, => @show_participants true + , 50 + + removed_from_view: -> + amplify.unsubscribe z.event.WebApp.SHORTCUT.PEOPLE + amplify.unsubscribe z.event.WebApp.SHORTCUT.ADD_PEOPLE + + click_on_call_button: => + amplify.publish z.event.WebApp.CALL.STATE.TOGGLE, @conversation_et().id + + click_on_participants: => + @show_participants() + + click_on_video_button: => + if @conversation_et().is_group() + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALL_NO_VIDEO_IN_GROUP + else + amplify.publish z.event.WebApp.CALL.STATE.TOGGLE, @conversation_et().id, true + + show_participants: (add_people) -> + amplify.publish z.event.WebApp.PEOPLE.TOGGLE, add_people diff --git a/app/script/view_model/DebugViewModel.coffee b/app/script/view_model/DebugViewModel.coffee new file mode 100644 index 00000000000..9333192b633 --- /dev/null +++ b/app/script/view_model/DebugViewModel.coffee @@ -0,0 +1,63 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.DebugViewModel + constructor: (element_id, @conversation_repository, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.DebugViewModel', z.config.LOGGER.OPTIONS + @debug_view = $("##{element_id}") + if @debug_view.length is 0 + return + + @title = ko.observable 'Debug info' + @info = ko.observable '' + @conversation = @conversation_repository.active_conversation + @self_user = @user_repository.self + + amplify.subscribe z.event.WebApp.SHORTCUT.DEBUG, -> $('#debug').toggleClass 'hide' + + ko.applyBindings @, @debug_view.get(0) + + inject_audio: (audio_file) -> + audio_file_path = "audio/buzzer/#{audio_file}" + return audio_file_path + + inject_audio_to_participants: -> + wire.app.repository.call_center.count_flows @conversation().id + + download_call_trace: -> + file_name = "#{wire.app.repository.user.self().name()} (#{wire.app.repository.user.self().id}).js" + text = JSON.stringify wire.app.repository.call_center.log_trace() + element = document.createElement 'a' + element.setAttribute 'href', "data:text/plain;charset=utf-8,#{encodeURIComponent(text)}" + element.setAttribute 'download', file_name + element.style.display = 'none' + document.body.appendChild element + element.click() + document.body.removeChild element + + log_call_to_console: -> + wire.app.repository.call_center.log_call() + + log_call_banner_state_to_console: -> + wire.app.view.content.call_controls.log_state() + + log_session_ids_to_console: -> + wire.app.repository.call_center.log_calls() diff --git a/app/script/view_model/GiphyViewModel.coffee b/app/script/view_model/GiphyViewModel.coffee new file mode 100644 index 00000000000..285dc941230 --- /dev/null +++ b/app/script/view_model/GiphyViewModel.coffee @@ -0,0 +1,143 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +NUMBER_OF_GIFS = 6 + +STATE = + DEFAULT: '' + ERROR: 'error' + LOADING: 'loading' + RESULTS: 'results' + +class z.ViewModel.GiphyViewModel + constructor: (@element_id, @conversation_repository, @giphy_repository) -> + @logger = new z.util.Logger 'z.ViewModel.GiphyViewModel', z.config.LOGGER.OPTIONS + + @modal = undefined + @state = ko.observable STATE.DEFAULT + @query = ko.observable '' + @show_giphy_button = ko.observable true + @sending_giphy_message = false + + # gif presented in the single gif view + @gif = ko.observable() + + # gifs rendered in the modal + @gifs = ko.observableArray() + + # gif selected by user or single gif when in single gif view + @selected_gif = ko.observable() + + @_init_subscriptions() + + _init_subscriptions: -> + amplify.subscribe z.event.WebApp.EXTENSIONS.GIPHY.SHOW, @show_giphy + + + show_giphy: => + @sending_giphy_message = false + @query @conversation_repository.active_conversation().input() + @state STATE.DEFAULT + @_get_random_gif() + @modal ?= new zeta.webapp.module.Modal '#giphy-modal' + @modal.show() + + + on_back: => + @gifs [@gif()] + @selected_gif @gif() + @show_giphy_button true + + on_try_another: => + @_get_random_gif() + + on_giphy_button: => + @_get_random_gifs() + + on_send: => + if @selected_gif() and not @sending_giphy_message + conversation_et = @conversation_repository.active_conversation() + @sending_giphy_message = true + @conversation_repository.send_gif conversation_et, @selected_gif().animated, @query(), -> + @sending_giphy_message = false + event = new z.tracking.event.PictureTakenEvent 'conversation', 'giphy', 'button' + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event.name, event.attributes + amplify.publish z.event.WebApp.EXTENSIONS.GIPHY.SEND + @modal.hide() + + on_close: => + @modal.hide() + + on_clicked_gif: (clicked_gif, event) => + return if @gifs().length is 1 + gif_item = $(event.currentTarget) + gif_items = gif_item.parent().children() + + remove_unselected = -> + $(@).removeClass 'gif-container-item-unselected' + + add_unselected = -> + $(@).addClass 'gif-container-item-unselected' + + if @selected_gif() is clicked_gif + gif_items.each remove_unselected + @selected_gif undefined + else + gif_items.each add_unselected + remove_unselected.apply gif_item + @selected_gif clicked_gif + + _clear_gifs: => + @gifs.removeAll() + @selected_gif undefined + @state STATE.LOADING + + _get_random_gif: => + return if @state() is STATE.ERROR + @_clear_gifs() + @show_giphy_button true + + @giphy_repository.get_random_gif + tag: @query() + .then (gif) => + @gif gif + @gifs.push @gif() + @selected_gif @gif() + @state STATE.RESULTS + .catch (error) => + @logger.log @logger.levels.ERROR, "No gif found for query: #{@query()}", error + @state STATE.ERROR + + _get_random_gifs: => + return if @state() is STATE.ERROR + @_clear_gifs() + @show_giphy_button false + + @giphy_repository.get_gifs + query: @query() + number: NUMBER_OF_GIFS + .then (gifs) => + @gifs gifs + @selected_gif(gifs[0]) if gifs.length is 1 + @state STATE.RESULTS + .catch (error) => + @logger.log @logger.levels.ERROR, "No gifs found for query: #{@query()}", error + @state STATE.ERROR diff --git a/app/script/view_model/ImageDetailViewViewModel.coffee b/app/script/view_model/ImageDetailViewViewModel.coffee new file mode 100644 index 00000000000..21d0ea27fc6 --- /dev/null +++ b/app/script/view_model/ImageDetailViewViewModel.coffee @@ -0,0 +1,73 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.ImageDetailViewViewModel + constructor: (@element_id) -> + + @image_element = undefined + @button_element = undefined + @image_modal = undefined + + amplify.subscribe z.event.WebApp.CONVERSATION.DETAIL_VIEW.SHOW, @show_detail_view + + show_detail_view: (src) => + element = $("##{@element_id}") + + @image_element = element.find '.detail-view-image' + @image_element[0].src = src + + @button_element = element.find '.detail-view-close-button' + + @image_modal.destroy() if @image_modal? + @image_modal = new zeta.webapp.module.Modal '#detail-view', @_hide_callback, @_before_hide_callback + @image_modal.show() + + @_show_image() + + hide_detail_view: => + @image_modal.hide() + + _before_hide_callback: => + @image_element.removeClass 'modal-content-anim-open' + + _hide_callback: => + $(window).off 'resize', @_center_image + + _show_image: => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.IMAGE_DETAIL_VIEW_OPENED + setTimeout => + @_center_image() + @image_element + .addClass 'modal-content-anim-open' + .one z.util.alias.animationend, => @_check_close_button() + , 0 + $(window).on 'resize', @_center_image + + _check_close_button: => + rect_image = @image_element[0].getBoundingClientRect() + rect_button = @button_element[0].getBoundingClientRect() + is_overlapping = rect_button.left < rect_image.right and rect_button.bottom > rect_image.top + @button_element.toggleClass 'detail-view-close-button-fullscreen', is_overlapping + + _center_image: => + @image_element.css + 'margin-left': (window.innerWidth - @image_element.width()) / 2 + 'margin-top': (window.innerHeight - @image_element.height()) / 2 diff --git a/app/script/view_model/LoadingViewModel.coffee b/app/script/view_model/LoadingViewModel.coffee new file mode 100644 index 00000000000..f42068a4991 --- /dev/null +++ b/app/script/view_model/LoadingViewModel.coffee @@ -0,0 +1,63 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.LoadingViewModel + constructor: (element_id, @user_repository) -> + @loading_message = ko.observable '' + @loading_step_current = ko.observable 0 + @loading_step_percentage = ko.observable 0 + @loading_step_total = 10 + + amplify.subscribe z.event.WebApp.APP.UPDATE_INIT, @switch_message + + ko.applyBindings @, document.getElementById element_id + + switch_message: (message_locator, next_step = false, replace_content) => + _create_message = (message_locator, replacements) -> + replacements = ({placeholder: replacement[0], content: replacement[1]} for replacement in replacements) + return z.localization.Localizer.get_text { + id: message_locator + replace: replacements + } + + switch message_locator + when z.string.init_received_self_user + message = _create_message message_locator, [['%name', @user_repository.self().first_name()]] + when z.string.init_events_expectation + if replace_content[0] > 200 + message_locator = z.string.init_events_expectation_long + message = _create_message message_locator, [['%events', replace_content[0]]] + when z.string.init_events_progress, z.string.init_sessions_progress + message = _create_message message_locator, [['%progress', replace_content[0]], ['%total', replace_content[1]]] + when z.string.init_sessions_expectation + if replace_content[0] > 100 + message_locator = z.string.init_sessions_expectation_long + message = _create_message message_locator, [['%sessions', replace_content[0]]] + else + message = z.localization.Localizer.get_text message_locator + + if not z.util.Environment.frontend.is_production() + @loading_message message + @_next_step() if next_step + + _next_step: -> + @loading_step_current (@loading_step_current() + 1) % @loading_step_total + @loading_step_percentage "#{@loading_step_current() / @loading_step_total * 100}%" diff --git a/app/script/view_model/MainViewModel.coffee b/app/script/view_model/MainViewModel.coffee new file mode 100644 index 00000000000..c14fbd3cedb --- /dev/null +++ b/app/script/view_model/MainViewModel.coffee @@ -0,0 +1,47 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.MainViewModel + constructor: (element_id, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.MainViewModel', z.config.LOGGER.OPTIONS + + @user = @user_repository.self + + @main_classes = ko.computed => + if @user()? + main_css_classes = "main-accent-color-#{@user().accent_id()}" # deprecated - still used on input control hover + main_css_classes += " #{@user().accent_theme()}" + return main_css_classes + + ko.applyBindings @, document.getElementById element_id + + amplify.subscribe z.event.WebApp.APP.HIDE, @_hide_app + amplify.subscribe z.event.WebApp.APP.FADE_IN, @_fade_in_app + + _hide_app: -> + $('#left, #right').css + opacity: 0 + 'pointer-events': 'none' + + _fade_in_app: -> + $('#left, #right').css + opacity: 1 + 'pointer-events': '' diff --git a/app/script/view_model/MessageListViewModel.coffee b/app/script/view_model/MessageListViewModel.coffee new file mode 100644 index 00000000000..42f2ccac14d --- /dev/null +++ b/app/script/view_model/MessageListViewModel.coffee @@ -0,0 +1,471 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +### +Message list rendering view model. + +@todo Get rid of the $('.conversation') opacity +@todo Get rid of the participants dependencies whenever bubble implementation has changed +@todo Remove all jquery selectors +### +class z.ViewModel.MessageListViewModel + constructor: (element_id, @conversation_repository, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.MessageListViewModel', z.config.LOGGER.OPTIONS + + @conversation = ko.observable new z.entity.Conversation() + @center_messages = ko.computed => + return not @conversation().has_further_messages() and @conversation().messages_visible().length is 1 and @conversation().messages_visible()[0].is_connection() + + @conversation_is_changing = false + + # is there a rendered message with an unread dot + @first_unread_timestamp = ko.observable() + + # store conversation to mark as read when browser gets focus + @mark_as_read_on_focus = undefined + + # can we used to prevent scroll handler from being executed (e.g. when using scrollTop()) + @capture_scrolling_event = false + + # store message subscription id + @messages_subscription = undefined + + # Last open bubble + @participant_bubble = undefined + @participant_bubble_last_id = undefined + + @viewport_changed = ko.observable false + @viewport_changed.extend rateLimit: 100 + + @recalculate_timeout = undefined + @on_initial_rendering = undefined + + @should_scroll_to_bottom = true + + # Check if the message container is to small and then pull new events + @on_mouse_wheel = _.throttle (e) => + is_not_scrollable = not $(e.currentTarget).is_scrollable() + is_scrolling_up = e.deltaY > 0 + if is_not_scrollable and is_scrolling_up + @_pull_events() + , 200 + + @on_scroll = _.throttle (data, e) => + return if not @capture_scrolling_event + + @viewport_changed not @viewport_changed() + + element = $ e.currentTarget + + # On some HDPI screen scrollTop returns a floating point number instead of an integer + # https://github.com/jquery/api.jquery.com/issues/608 + scroll_position = Math.ceil element.scrollTop() + scroll_end = element.scroll_end() + scrolled_bottom = false + + if scroll_position is 0 + @_pull_events() + + if scroll_position >= scroll_end + scrolled_bottom = true + + if document.hasFocus() + @conversation_repository.mark_as_read @conversation() + else + @mark_as_read_on_focus = @conversation() + + @should_scroll_to_bottom = scroll_position > scroll_end - z.config.SCROLL_TO_LAST_MESSAGE_THRESHOLD + + amplify.publish z.event.WebApp.LIST.SCROLL, scrolled_bottom + , 100 + + $(window) + .on 'resize', => + @viewport_changed not @viewport_changed() + .on 'focus', => + if @mark_as_read_on_focus? + window.setTimeout => + @conversation_repository.mark_as_read @mark_as_read_on_focus + @mark_as_read_on_focus = undefined + , 1000 + + amplify.subscribe z.event.WebApp.CONVERSATION.PEOPLE.HIDE, @hide_bubble + + ### + Remove all subscriptions and reset states. + @param conversation_et [z.entity.Conversation] Conversation entity to change to + ### + release_conversation: (conversation_et) => + conversation_et?.release() + @messages_subscription?.dispose() + @capture_scrolling_event = false + @first_unread_timestamp undefined + + ### + Change conversation. + @param conversation_et [z.entity.Conversation] Conversation entity to change to + @param callback [Function] Executed when all events are loaded an conversation is ready to be displayed + ### + change_conversation: (conversation_et, callback) => + @conversation_is_changing = true + + # clean up old conversation + @release_conversation @conversation() if @conversation() + + # update new conversation + @conversation conversation_et + + if not conversation_et.is_loaded() + @conversation_repository.update_participating_user_ets conversation_et, (conversation_et) => + @conversation_repository.get_events conversation_et + .then => + @_set_conversation conversation_et, callback + conversation_et.is_loaded true + else + @_set_conversation conversation_et, callback + + ### + Sets the conversation and waits for further processing until knockout has rendered the messages. + @param conversation_et [z.entity.Conversation] Conversation entity to set + @param callback [Function] Executed when message list is ready to fade in + ### + _set_conversation: (conversation_et, callback) => + # hide conversation until everything is processed + $('.conversation').css opacity: 0 + + @conversation_is_changing = false + + if @conversation().messages_visible().length is 0 + # return immediately if nothing to render + @_initial_rendering conversation_et, callback + else + # will be executed after all messages are rendered + @on_initial_rendering = _.once => @_initial_rendering conversation_et, callback + + ### + Registers for mouse wheel events and incoming messages. + + @note Call this once after changing conversation. + @param conversation_et [z.entity.Conversation] Conversation entity to render + @param callback [Function] Executed when message list is ready to fade in + ### + _initial_rendering: (conversation_et, callback) => + messages_container = $('.messages-wrap') + messages_container.on 'mousewheel', @on_mouse_wheel + + window.requestAnimFrame => + is_current_conversation = conversation_et is @conversation() + if not is_current_conversation + @logger.log @logger.levels.INFO, 'Skipped loading conversation', conversation_et.display_name() + return + + # reset scroll position + messages_container.scrollTop 0 + + @capture_scrolling_event = true + + if not messages_container.is_scrollable() + @conversation_repository.mark_as_read conversation_et + else + unread_message = $ '.message-timestamp-unread' + if unread_message.length > 0 + messages_container.scroll_by unread_message.parent().parent().position().top + else + messages_container.scroll_to_bottom() + $('.conversation').css opacity: 1 + + # subscribe for incoming messages + @messages_subscription = conversation_et.messages_visible.subscribe @_on_message_add, null, 'arrayChange' + @_subscribe_to_iframe_clicks() + callback?() + + ### + Checks how to scroll message list and if conversation should be marked as unread. + + @param message [Array] Array of message entities + ### + _on_message_add: (messages) => + messages_container = $('.messages-wrap') + last_item = messages[messages.length - 1] + last_message = last_item.value + + # we are only interested in items that were added + if last_item.status isnt 'added' + return + + # message was prepended + if last_message?.timestamp isnt @conversation().last_event_timestamp() + return + + # scroll to bottom if self user send the message + if last_message?.from is @user_repository.self().id + window.requestAnimFrame -> messages_container.scroll_to_bottom() + return + + # scroll to the end of the list if we are under a certain threshold + if @should_scroll_to_bottom + @conversation_repository.mark_as_read @conversation() if document.hasFocus() + window.requestAnimFrame -> messages_container.scroll_to_bottom() + + # mark as read when conversation is not scrollable + is_scrollable = messages_container.is_scrollable() + is_browser_has_focus = document.hasFocus() + if not is_scrollable + if is_browser_has_focus + @conversation_repository.mark_as_read @conversation() + else + @mark_as_read_on_focus = @conversation() + + # Get previous messages from the backend. + _pull_events: => + if not @conversation().is_pending() and @conversation().has_further_messages() + inner_container = $('.messages-wrap').children()[0] + old_list_height = inner_container.scrollHeight + + @capture_scrolling_event = false + @conversation_repository.get_events @conversation() + .then => + new_list_height = inner_container.scrollHeight + $('.messages-wrap').scrollTop new_list_height - old_list_height + @capture_scrolling_event = true + + scroll_height: (change_in_height) -> + $('.messages-wrap').scroll_by change_in_height + + ### + Triggered when user clicks on an avatar in the message list. + @param user_et [z.entity.User] User entity of the selected user + @param message [DOMElement] Selected DOMElement + ### + on_message_user_click: (user_et, element) => + BUBBLE_HEIGHT = 440 + MESSAGE_LIST_MIN_HEIGHT = 400 + list_height = $('.message-list').height() + element_rect = element.getBoundingClientRect() + element_distance_top = element_rect.top + element_distance_bottom = list_height - element_rect.top - element_rect.height + largest_distance = Math.max element_distance_top, element_distance_bottom + difference = BUBBLE_HEIGHT - largest_distance + + create_bubble = (element_id) => + wire.app.view.content.participants.reset_view() + @participant_bubble_last_id = element_id + @participant_bubble = new zeta.webapp.module.Bubble + host_selector: "##{element_id}" + scroll_selector: '.messages-wrap' + modal: true + on_show: -> + amplify.publish z.event.WebApp.PEOPLE.SHOW, user_et + on_hide: => + @participant_bubble = undefined + @participant_bubble_last_id = undefined + @participant_bubble.toggle() + + show_bubble = => + wire.app.view.content.participants.confirm_dialog?.destroy() + # we clicked on the same bubble + if @participant_bubble and @participant_bubble_last_id is element.id + @participant_bubble.toggle() + return + + # dismiss old bubble and wait with creating the new one when another bubble is open + if @participant_bubble or wire.app.view.content.participants.participants_bubble?.is_visible() + @participant_bubble?.hide() + window.setTimeout -> + create_bubble(element.id) + , 550 + else + create_bubble(element.id) + + if difference > 0 and list_height > MESSAGE_LIST_MIN_HEIGHT + if largest_distance is element_distance_top + @scroll_by -difference, show_bubble + else + @scroll_by difference, show_bubble + else + show_bubble() + + ### + Triggered when user clicks on the session reset link in a decrypt error message. + @param message_et [z.entity.DecryptErrorMessage] Decrypt error message + ### + on_session_reset_click: (message_et) => + reset_progress = -> + window.setTimeout -> + message_et.is_resetting_session false + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.SESSION_RESET + , 550 + + message_et.is_resetting_session true + @conversation_repository.reset_session message_et.from, message_et.client_id, @conversation().id + .then -> reset_progress() + .catch -> reset_progress() + + ### + Is called by knockout whenever a new message is rendered. + @param elements [Array] List of elements that were added to the DOM (Note: This also contains html comments) + @param message [z.entity.Message] rendered message + ### + after_message_render: (elements, message) => + window.requestAnimFrame => + message_index = @conversation_repository.active_conversation().messages_visible().indexOf message + if message_index > 0 + last_message = @conversation_repository.active_conversation().messages_visible()[message_index - 1] + + last = moment.unix last_message?.timestamp / 1000 + current = moment.unix message.timestamp / 1000 + + if not last_message? or moment(current).diff(last, 'minutes') > 60 and message.is_content() + $(elements) + .find '.message-timestamp' + .removeClass 'message-timestamp-hidden' + + if last_message? + ### + @note For content messages (except pings): + Don't show user avatar next to message if the last message in the conversation was already sent by the same user + ### + if last_message.is_content() and last_message.user().id is message.user().id and message.is_content() + $(elements) + .find '.message-header-user-is-hideable' + .addClass 'hide-user' + + if last_message.timestamp is @conversation().last_read_timestamp() and not @first_unread_timestamp() + @first_unread_timestamp message.timestamp + $(elements) + .find '.message-timestamp' + .addClass 'message-timestamp-unread' + .removeClass 'message-timestamp-hidden' + .end() + .find '.message-header-user-is-hideable' + .removeClass 'hide-user' + + if not last.isSame current, 'day' + $(elements) + .find '.message-timestamp' + .addClass 'message-timestamp-day' + .removeClass 'message-timestamp-hidden' + .end() + .find '.message-header-user-is-hideable' + .removeClass 'hide-user' + + if message?.is_ping() + now = Date.now() + message.animated now - current < 2000 + + if z.util.array_is_last @conversation().messages_visible(), message + # Defer initial rendering + window.requestAnimFrame => @on_initial_rendering() + + before_message_remove: (dom_node) -> + if $(dom_node).hasClass 'message' and not @conversation_is_changing + + has_timestamp = $(dom_node).find('.message-timestamp-hidden').length is 0 + has_day_timestamp = $(dom_node).find('.message-timestamp-day').length > 0 + has_avatar = $(dom_node).find('.hide-user').length is 0 + next_message = $(dom_node).next() + + if has_timestamp + next_message + .find '.message-timestamp' + .removeClass 'message-timestamp-hidden' + .end() + .find '.message-header-user-is-hideable' + .removeClass 'hide-user' + else if has_day_timestamp + next_message + .find '.message-timestamp' + .addClass 'message-timestamp-day' + .removeClass 'message-timestamp-hidden' + .end() + .find '.message-header-user-is-hideable' + .removeClass 'hide-user' + else if has_avatar + next_message + .find '.message-header-user-is-hideable' + .removeClass 'hide-user' + + $(dom_node) + .addClass 'message-fade-out' + .on 'transitionend', -> + $(@).remove() + else + # clean up whatever this is + $(dom_node).remove() + + # Subscribes to iFrame click events. + _subscribe_to_iframe_clicks: -> + $('iframe.soundcloud').iframeTracker blurCallback: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.SOUNDCLOUD_CONTENT_CLICKED + + $('iframe.youtube').iframeTracker blurCallback: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.YOUTUBE_CONTENT_CLICKED + + # Hides participant bubble. + hide_bubble: => + @participant_bubble?.hide() + + ### + Scrolls whole message list by given distance. + + @note Scrolling is animated with jQuery + @param distance [Number] Distance by which the container is shifted + @param callback [Function] Executed when scroll animation is finished + ### + scroll_by: (distance, callback) -> + current_scroll = $('.messages-wrap').scrollTop() + new_scroll = current_scroll + distance + $('.messages-wrap').animate {scrollTop: new_scroll}, 300, callback + + ### + Gets CSS class that will be applied to the message div in order to style. + @param message [z.entity.Message] Message entity for generating css class + @return [String] CSS class that is applied to the element + ### + get_css_class: (message) -> + switch message.super_type + when z.message.SuperType.CALL + return 'message-system message-call' + when z.message.SuperType.CONTENT + return 'message-normal' + when z.message.SuperType.MEMBER + return 'message message-system message-member' + when z.message.SuperType.PING + return 'message-ping' + when z.message.SuperType.SYSTEM + if message.system_message_type is z.message.SystemMessageType.CONVERSATION_RENAME + return 'message-system message-rename' + when z.message.SuperType.UNABLE_TO_DECRYPT + return 'message-system' + + ### + Shows detail image view. + @param asset_et [z.assets.Asset] Asset to be displayed + @param event [UIEvent] Actual scroll event + ### + show_detail: (asset_et, event) -> + target_element = $(event.currentTarget) + return if target_element.hasClass 'image-loading' + amplify.publish z.event.WebApp.CONVERSATION.DETAIL_VIEW.SHOW, target_element.find('img')[0].src + + click_on_cancel_request: (message_et) => + next_conversation_et = @conversation_repository.get_next_conversation @conversation_repository.active_conversation() + @user_repository.cancel_connection_request message_et.other_user(), next_conversation_et diff --git a/app/script/view_model/ModalsViewModel.coffee b/app/script/view_model/ModalsViewModel.coffee new file mode 100644 index 00000000000..d1afe912d56 --- /dev/null +++ b/app/script/view_model/ModalsViewModel.coffee @@ -0,0 +1,210 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +z.ViewModel.ModalType = + BLOCK: '.modal-block' + CALLING: '.modal-calling' + CALL_EMPTY_CONVERSATION: '.modal-call-conversation-empty' + CALL_FULL_CONVERSATION: '.modal-call-conversation-full' + CALL_FULL_VOICE_CHANNEL: '.modal-call-voice-channel-full' + CALL_NO_VIDEO_IN_GROUP: '.modal-call-no-video-in-group' + CALL_START_ANOTHER: '.modal-call-second' + CLEAR: '.modal-clear' + CLEAR_GROUP: '.modal-clear-group' + CONNECTED_DEVICE: '.modal-connected-device' + CONTACTS: '.modal-contacts' + DELETE_MESSAGE: '.modal-delete-message' + TOO_LONG_MESSAGE: '.modal-too-long-message' + TOO_MANY_MEMBERS: '.modal-too-many-members' + LEAVE: '.modal-leave' + LOGOUT: '.modal-logout' + NEW_DEVICE: '.modal-new-device' + SESSION_RESET: '.modal-session-reset' + UPLOAD_PARALLEL: '.modal-asset-upload-parallel' + UPLOAD_TOO_LARGE: '.modal-asset-upload-too-large' + WHITELIST_SCREENSHARING: '.modal-whitelist-screensharing' + +class z.ViewModel.ModalsViewModel + constructor: (element_id) -> + @logger = new z.util.Logger 'z.ViewModel.ModalsViewModel', z.config.LOGGER.OPTIONS + + @modals = {} + + amplify.subscribe z.event.WebApp.WARNINGS.MODAL, @show_modal + + ko.applyBindings @, document.getElementById element_id + + ### + Show modal + + @param type [z.ViewModel.ModalType] Indicates which modal to show + @param options [Object] + @option data [Object] Content needed for visualization on modal + @option action [Function] Function to be called when action in modal is triggered + ### + show_modal: (type, options = {}) => + message_element = $(type).find('.modal-text') + title_element = $(type).find('.modal-title') + switch type + when z.ViewModel.ModalType.BLOCK + @_show_modal_block options.data, title_element, message_element + when z.ViewModel.ModalType.CALL_FULL_CONVERSATION + @_show_modal_call_full_conversation options.data, message_element + when z.ViewModel.ModalType.CALL_FULL_VOICE_CHANNEL + @_show_modal_call_full_voice_channel options.data, message_element + when z.ViewModel.ModalType.CALL_START_ANOTHER + @_show_modal_call_start_another options.data, title_element, message_element + when z.ViewModel.ModalType.CLEAR + type = @_show_modal_clear options, type + when z.ViewModel.ModalType.CONNECTED_DEVICE + @_show_modal_connected_device options.data + when z.ViewModel.ModalType.LEAVE + @_show_modal_leave options.data, title_element + when z.ViewModel.ModalType.NEW_DEVICE + @_show_modal_new_device options.data, title_element + when z.ViewModel.ModalType.TOO_MANY_MEMBERS + @_show_modal_too_many_members options.data, message_element + when z.ViewModel.ModalType.UPLOAD_PARALLEL + @_show_modal_upload_parallel options.data, title_element + when z.ViewModel.ModalType.UPLOAD_TOO_LARGE + @_show_modal_upload_too_large options.data, title_element + when z.ViewModel.ModalType.TOO_LONG_MESSAGE + @_show_modal_message_too_long options.data, message_element + + modal = new zeta.webapp.module.Modal type, null, -> + $(type).find('.modal-close').off 'click' + $(type).find('.modal-action').off 'click' + $(type).find('.modal-secondary').off 'click' + modal.destroy() + options.close?() + + $(type).find('.modal-close').click -> + modal.hide() + + $(type).find('.modal-secondary').click -> + modal.hide -> options.secondary?() + + $(type).find('.modal-action').click -> + modal.hide -> + checkbox = $(type).find('.modal-option-checkbox') + if checkbox + options.action checkbox.is ':checked' + checkbox.attr 'checked', false + else + options.action?() + + @logger.log @logger.levels.INFO, "Toggle modal of type '#{type}'" + modal.toggle() + + _show_modal_block: (content, title_element, message_element) -> + title_element.text z.localization.Localizer.get_text { + id: z.string.modal_block_conversation_headline + replace: {placeholder: '%@.name', content: content} + } + message_element.text z.localization.Localizer.get_text { + id: z.string.modal_block_conversation_message + replace: {placeholder: '%@.name', content: content} + } + + _show_modal_call_full_conversation: (content, message_element) -> + message_element.text z.localization.Localizer.get_text { + id: z.string.modal_call_conversation_full_message + replace: {placeholder: '%no', content: content} + } + + _show_modal_call_full_voice_channel: (content, message_element) -> + message_element.text z.localization.Localizer.get_text { + id: z.string.modal_call_voice_channel_full_message + replace: {placeholder: '%no', content: content} + } + + _show_modal_call_start_another: (is_outgoing, title_element, message_element) -> + action_element = $(z.ViewModel.ModalType.CALL_START_ANOTHER).find('.modal-action') + if is_outgoing + action_element.text z.localization.Localizer.get_text z.string.modal_call_second_outgoing_action + message_element.text z.localization.Localizer.get_text z.string.modal_call_second_outgoing_message + title_element.text z.localization.Localizer.get_text z.string.modal_call_second_outgoing_headline + else + action_element.text z.localization.Localizer.get_text z.string.modal_call_second_incoming_action + message_element.text z.localization.Localizer.get_text z.string.modal_call_second_incoming_message + title_element.text z.localization.Localizer.get_text z.string.modal_call_second_incoming_headline + + _show_modal_clear: (options, type) -> + if options.conversation.is_group() and not options.conversation.removed_from_conversation() + type = z.ViewModel.ModalType.CLEAR_GROUP + + title_element = $(type).find('.modal-title') + title_element.text z.localization.Localizer.get_text { + id: z.string.modal_clear_conversation_headline + replace: {placeholder: '%@.name', content: options.data} + } + + return type + + _show_modal_connected_device: (devices) -> + devices_element = $(z.ViewModel.ModalType.CONNECTED_DEVICE).find('.modal-connected-devices') + devices_element.empty() + for device in devices + $('
    ') + .text "#{moment(device.time).format 'MMMM Do YYYY, HH:mm'} - UTC" + .appendTo devices_element + $('
    ') + .text "#{z.localization.Localizer.get_text z.string.modal_connected_device_from} #{device.model}" + .appendTo devices_element + + _show_modal_leave: (content, title_element) -> + title_element.text z.localization.Localizer.get_text { + id: z.string.modal_leave_conversation_headline + replace: {placeholder: '%@.name', content: content} + } + + _show_modal_new_device: (content, title_element) -> + title_element.text z.localization.Localizer.get_text { + id: z.string.modal_new_device_headline + replace: {placeholder: '%@.name', content: content} + } + + _show_modal_too_many_members: (content, message_element) -> + message_element.text z.localization.Localizer.get_text { + id: z.string.modal_too_many_members_message + replace: [ + {placeholder: '%no', content: content.open_spots} + {placeholder: '%max', content: content.max} + ] + } + + _show_modal_upload_parallel: (content, title_element) -> + title_element.text z.localization.Localizer.get_text { + id: z.string.modal_uploads_parallel + replace: {placeholder: '%no', content: content} + } + + _show_modal_upload_too_large: (content, title_element) -> + title_element.text z.localization.Localizer.get_text { + id: z.string.conversation_asset_upload_too_large + replace: {placeholder: '%no', content: content} + } + + _show_modal_message_too_long: (content, message_element) -> + message_element.text z.localization.Localizer.get_text { + id: z.string.modal_too_long_message + replace: {placeholder: '%no', content: content} + } diff --git a/app/script/view_model/ParticipantsViewModel.coffee b/app/script/view_model/ParticipantsViewModel.coffee new file mode 100644 index 00000000000..07adc25d983 --- /dev/null +++ b/app/script/view_model/ParticipantsViewModel.coffee @@ -0,0 +1,238 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +# participants view state +STATE = + PARTICIPANTS: 'participants' + SEARCH: 'search' + +class z.ViewModel.ParticipantsViewModel + constructor: (@element_id, @user_repository, @conversation_repository, @search_repository) -> + @logger = new z.util.Logger 'z.ViewModel.ParticipantsViewModel', z.config.LOGGER.OPTIONS + + @state = ko.observable STATE.PARTICIPANTS + + @conversation = ko.observable new z.entity.Conversation() + @conversation.subscribe => @render_participants false + + @render_participants = ko.observable false + + @participants = ko.observableArray() + @participants_verified = ko.observableArray() + @participants_unverified = ko.observableArray() + + ko.computed => + conversation_et = @conversation() + participants = [].concat conversation_et.participating_user_ets() + participants.sort z.util.sort_user_by_first_name + + @participants participants + @participants_verified (user_et for user_et in participants when user_et.is_verified()) + @participants_unverified (user_et for user_et in participants when not user_et.is_verified()) + + # confirm dialog reference + @confirm_dialog = undefined + + # selected group user + @user_profile = ko.observable new z.entity.User() + + # switch between div and input field to edit the conversation name + @editing = ko.observable false + @edit = -> @editing true + + @editing.subscribe (value) => + if value is false + name = $('.group-header .name span') + $('.group-header textarea').css('height', "#{name.height()}px") + else + $('.group-header textarea').val(@conversation().display_name()) + + @participants_bubble = new zeta.webapp.module.Bubble + host_selector: '#show-participants' + scroll_selector: '.messages-wrap' + modal: true + on_hide: => @reset_view() + + # TODO create a viewmodel search? + @user_input = ko.observable '' + @user_selected = ko.observableArray [] + @connected_users = ko.computed => + connected_users = ko.utils.arrayFilter @user_repository.users(), (user_et) => + is_participant = ko.utils.arrayFirst @participants(), (participant) -> user_et.id is participant.id + is_connected = user_et.connection().status() is z.user.ConnectionStatus.ACCEPTED + return is_participant is null and is_connected + connected_users.sort z.util.sort_user_by_first_name + , @, deferEvaluation: true + + @add_people_tooltip = z.localization.Localizer.get_text { + id: z.string.tooltip_people_add + replace: {placeholder: '%shortcut', content: z.ui.Shortcut.get_shortcut_tooltip z.ui.ShortcutType.ADD_PEOPLE} + } + + amplify.subscribe z.event.WebApp.PENDING.SHOW, => + @participants_bubble.hide() + + amplify.subscribe z.event.WebApp.PEOPLE.SHOW, (user_et) => + @user_profile user_et + $("##{@element_id}").addClass 'single-user-mode' + + toggle_participants_bubble: (add_people = false) => + toggle_bubble = => + if not @participants_bubble.is_visible() + @reset_view() + + if @conversation().is_group() + @user_profile new z.entity.User() + else + @user_profile @participants()[0] + + @render_participants true + $("##{@element_id}").removeClass 'single-user-mode' + + if add_people + if not @participants_bubble.is_visible() + @participants_bubble.show() + @add_people() + else if @state() is STATE.SEARCH or @confirm_dialog?.is_visible() + @participants_bubble.hide() + else + @add_people() + else + @participants_bubble.toggle() + + if wire.app.view.content.message_list.participant_bubble?.is_visible() + setTimeout -> + toggle_bubble() + , 550 + else + toggle_bubble() + + change_conversation: (conversation_et) -> + @participants_bubble.hide() + @conversation conversation_et + @user_profile new z.entity.User() + + reset_view: => + @state STATE.PARTICIPANTS + @user_selected.removeAll() + @confirm_dialog?.destroy() + $("##{@element_id}").removeClass 'single-user-mode' + + add_people: => + @state STATE.SEARCH + $('.participants-search').addClass 'participants-search-show' + + leave_conversation: => + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-leave' + confirm: => + next_conversation_et = @conversation_repository.get_next_conversation @conversation() + @participants_bubble.hide() + @conversation_repository.leave_conversation @conversation(), next_conversation_et + + show_participant: (user_et) => + @user_profile user_et + + rename_conversation: (data, event) => + new_name = z.util.remove_line_breaks event.target.value.trim() + old_name = @conversation().display_name().trim() + event.target.value = old_name + @editing false + if new_name.length > 0 and new_name isnt old_name + @conversation_repository.rename_conversation @conversation(), new_name + + on_search_add: => + user_ids = ko.utils.arrayMap @user_selected(), (user_et) -> user_et.id + @participants_bubble.hide() + + if @conversation().is_group() + @conversation_repository.add_members @conversation(), user_ids, => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.ADD_TO_GROUP_CONVERSATION, + {numberOfParticipantsAdded: user_ids.length, numberOfGroupParticipants: @conversation().number_of_participants()} + else + user_ids = user_ids.concat @user_profile().id + @conversation_repository.create_new_conversation user_ids, null, (conversation_et) -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.CREATE_GROUP_CONVERSATION, + {creationContext: 'addedToOneToOne', numberOfParticipants: user_ids.length} + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + + on_search_close: => + @reset_view() + + close: => + @user_profile new z.entity.User() + @reset_view() + + remove: (user_et) => + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-remove' + data: + user: user_et + confirm: => + @conversation_repository.remove_member @conversation(), user_et.id, (response, error) => + @reset_view() if response + + show_self_profile: -> + amplify.publish z.event.WebApp.PROFILE.SHOW + + unblock: (user_et) => + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-unblock' + data: + user: user_et + confirm: => + @user_repository.unblock_user user_et + .then => + @participants_bubble.hide() + conversation_et = @conversation_repository.get_one_to_one_conversation user_et.id + @conversation_repository.update_participating_user_ets conversation_et + + block: (user_et) => + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-block' + data: + user: user_et + confirm: => + next_conversation_et = @conversation_repository.get_next_conversation @conversation() + @participants_bubble.hide() + @user_repository.block_user user_et + .then => + amplify.publish z.event.WebApp.CONVERSATION.SWITCH, @conversation(), next_conversation_et + + connect: (user_et) => + @participants_bubble.hide() + + pending: (user_et) => + on_success = => @participants_bubble.hide() + + @confirm_dialog = $('#participants').confirm + template: '#template-confirm-connect' + data: + user: @user_profile() + confirm: => + @user_repository.accept_connection_request user_et, true + .then -> on_success() + cancel: => + @user_repository.ignore_connection_request user_et + .then -> on_success() diff --git a/app/script/view_model/RightViewModel.coffee b/app/script/view_model/RightViewModel.coffee new file mode 100644 index 00000000000..97d00d6889d --- /dev/null +++ b/app/script/view_model/RightViewModel.coffee @@ -0,0 +1,176 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +z.ViewModel.CONTENT_STATE = + PENDING: 'pending' + CONVERSATION: 'conversation' + PROFILE: 'profile' + BLANK: '' + + +class z.ViewModel.RightViewModel + constructor: (element_id, @user_repository, @conversation_repository, @call_repository, @search_repository, @giphy_repository, @client_repository) -> + @logger = new z.util.Logger 'z.ViewModel.RightViewModel', z.config.LOGGER.OPTIONS + + # state + @state = ko.observable z.ViewModel.CONTENT_STATE.BLANK + + # nested view models + @call_shortcuts = new z.ViewModel.CallShortcutsViewModel @call_repository + @video_calling = new z.ViewModel.VideoCallingViewModel 'video-calling', @call_repository, @user_repository, @conversation_repository + @connect_requests = new z.ViewModel.ConnectRequestsViewModel 'connect-requests', @user_repository + @conversation_titlebar = new z.ViewModel.ConversationTitlebarViewModel 'conversation-titlebar', @conversation_repository + @conversation_input = new z.ViewModel.ConversationInputViewModel 'conversation-input', @conversation_repository, @user_repository + @message_list = new z.ViewModel.MessageListViewModel 'message-list', @conversation_repository, @user_repository + @participants = new z.ViewModel.ParticipantsViewModel 'participants', @user_repository, @conversation_repository, @search_repository + @self_profile = new z.ViewModel.SelfProfileViewModel 'self-profile', @user_repository, @client_repository + @giphy = new z.ViewModel.GiphyViewModel 'giphy-modal', @conversation_repository, @giphy_repository + @detail_view = new z.ViewModel.ImageDetailViewViewModel 'detail-view' + + @previous_state = undefined + @previous_conversation = undefined + + @state.subscribe (value) => + if value is z.ViewModel.CONTENT_STATE.CONVERSATION + @conversation_input.added_to_view() + @conversation_titlebar.added_to_view() + else + @conversation_input.removed_from_view() + @conversation_titlebar.removed_from_view() + + @user_repository.connect_requests.subscribe (requests) => + if @state() is z.ViewModel.CONTENT_STATE.PENDING and requests.length is 0 + @show_conversation @conversation_repository.get_most_recent_conversation() + + @_init_subscriptions() + + ko.applyBindings @, document.getElementById element_id + + _init_subscriptions: => + amplify.subscribe z.event.WebApp.CONVERSATION.SHOW, @show_conversation + amplify.subscribe z.event.WebApp.CONVERSATION.SWITCH, @switch_conversation + amplify.subscribe z.event.WebApp.LIST.SCROLL, @conversation_input.show_separator + amplify.subscribe z.event.WebApp.PENDING.SHOW, @show_connect_requests + amplify.subscribe z.event.WebApp.PEOPLE.TOGGLE, @participants.toggle_participants_bubble + amplify.subscribe z.event.WebApp.PROFILE.SHOW, @show_self_profile + amplify.subscribe z.event.WebApp.PROFILE.HIDE, @hide_self_profile + amplify.subscribe z.event.WebApp.WINDOW.RESIZE.HEIGHT, @message_list.scroll_height + + ### + Slide in specified content. + + @param content_selector [String] dom element to apply slide in animation + ### + _shift_content: (content_selector) -> + incoming_css_class = 'content-animation-incoming' + + $(content_selector) + .removeClass incoming_css_class + .off z.util.alias.animationend + .addClass incoming_css_class + .one z.util.alias.animationend, -> + $(@).removeClass(incoming_css_class).off z.util.alias.animationend + + ### + Opens the specified conversation. + + @note If the conversation_et is not defined, it will open the incoming connection requests instead + Conversation_et can also just be the conversation ID + + @param conversation_et [z.entity.Conversation | String] Conversation entity or conversation ID + ### + show_conversation: (conversation_et) => + return @show_connect_requests() if not conversation_et? + conversation_et = @conversation_repository.get_conversation_by_id conversation_et if not conversation_et.id? + return if conversation_et is @conversation_repository.active_conversation() + + show_conversation = => + @logger.log @logger.levels.LEVEL_1, "Switching view to conversation: #{conversation_et.id}" + @state z.ViewModel.CONTENT_STATE.CONVERSATION + @message_list.change_conversation conversation_et, => + @_shift_content '.conversation' + @participants.change_conversation conversation_et + + @conversation_repository.active_conversation conversation_et + + if @state() is z.ViewModel.CONTENT_STATE.PROFILE + @self_profile.hide() + setTimeout show_conversation, 750 # wait for self profile to disappear + else + show_conversation() + + ### + Opens the incoming connection requests. + + @note If there are no connection requests, it will open the self profile instead + ### + show_connect_requests: => + return @show_self_profile() if @user_repository.connect_requests().length < 1 + + show_connect_request = => + @conversation_repository.active_conversation null + @state z.ViewModel.CONTENT_STATE.PENDING + @_shift_content '.connect-requests' + + @message_list.release_conversation() if @state() is z.ViewModel.CONTENT_STATE.CONVERSATION + + if @state() is z.ViewModel.CONTENT_STATE.PROFILE + @self_profile.hide() + setTimeout show_connect_request, 750 # wait for self profile to disappear + else + show_connect_request() + + ### + Open self profile. + + @param animate [Boolean] Do background animation + ### + show_self_profile: (animate = true) => + return if @state() is z.ViewModel.CONTENT_STATE.PROFILE + + @previous_state = @state() + @previous_conversation = @conversation_repository.active_conversation() + + @message_list.release_conversation() if @state() is z.ViewModel.CONTENT_STATE.CONVERSATION + + @conversation_repository.active_conversation null + amplify.publish z.event.WebApp.LIST.FULLSCREEN_ANIM_DISABLED if not animate + @state z.ViewModel.CONTENT_STATE.PROFILE + @self_profile.show() + + ### + Close self profile. + ### + hide_self_profile: => + if @previous_state is z.ViewModel.CONTENT_STATE.PENDING + @show_connect_requests() + else + @show_conversation @previous_conversation + + ### + Switches the conversation if the other one is shown. + + @param conversation_et [z.entity.Conversation] Conversation entity to be verified as currently active for the switch + @param next_conversation_et [z.entity.Conversation] Conversation entity to be shown + ### + switch_conversation: (conversation_et, next_conversation_et) => + @show_conversation next_conversation_et if @conversation_repository.is_active_conversation conversation_et diff --git a/app/script/view_model/SelfProfileViewModel.coffee b/app/script/view_model/SelfProfileViewModel.coffee new file mode 100644 index 00000000000..920ae66d18d --- /dev/null +++ b/app/script/view_model/SelfProfileViewModel.coffee @@ -0,0 +1,165 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +SETTING = + ALL: '0' + NONE: '2' + SOME: '1' + +LOCALYTICS_SOUND_SETTING = + ALL: 'alwaysPlay' + SOME: 'FirstMessageOnly' + NONE: 'neverPlay' + +class z.ViewModel.SelfProfileViewModel + constructor: (@element_id, @user_repository, @client_repository) -> + @logger = new z.util.Logger 'z.ViewModel.SelfProfileViewModel', z.config.LOGGER.OPTIONS + + @user = @user_repository.self + @settings_bubble = new zeta.webapp.module.Bubble host_selector: '#show-settings' + + @new_clients = ko.observableArray() + + $('.self-profile').on z.util.alias.animationend, -> + profile = $(@) + profile.removeClass 'self-profile-transition-in self-profile-transition-out' + profile.hide() if profile.hasClass 'self-profile-hidden' + + amplify.subscribe z.event.WebApp.LOGOUT.ASK_TO_CLEAR_DATA, @logout + amplify.subscribe z.event.WebApp.PROFILE.UPLOAD_PICTURE, @set_picture + amplify.subscribe z.event.WebApp.SELF.CLIENT_ADD, @on_client_add + +############################################################################### +# Self Profile +############################################################################### + + change_accent_color: (color) => + @user_repository.change_accent_color color.id + + change_username: (name) => + @user_repository.change_username name + + hide: -> + $('.self-profile') + .addClass 'self-profile-hidden self-profile-transition-out' + .removeClass 'self-profile-visible' + + show: -> + $('.self-profile').show() + + window.setTimeout -> + $('.self-profile') + .addClass 'self-profile-visible self-profile-transition-in' + .removeClass 'self-profile-hidden' + , 17 + + if @new_clients().length + setTimeout @on_show_new_clients, 1000 + + toggle_settings_menu: -> + @settings_bubble.toggle() + +############################################################################### +# Settings menu +############################################################################### + + logout: => + # TODO: Rely on client repository + if @client_repository.current_client().type is z.client.ClientType.PERMANENT + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.LOGOUT, + action: (clear_data) -> + amplify.publish z.event.WebApp.SIGN_OUT, 'user_requested', clear_data + else + @client_repository.delete_temporary_client() + .then -> amplify.publish z.event.WebApp.SIGN_OUT, 'user_requested', true + + show_support_page: -> + (window.open z.string.url_support)?.focus() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SETTINGS_MENU.SHOW_SUPPORT_PAGE + + toggle_about: -> + @about_modal ?= new zeta.webapp.module.Modal '#self-about' + @about_modal.toggle() + if @about_modal.is_shown() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SETTINGS_MENU.SHOW_ABOUT_SCREEN + @settings_bubble.hide() + + toggle_settings: -> + @settings_bubble.hide() + amplify.publish z.event.WebApp.PROFILE.SETTINGS.SHOW + +############################################################################### +# Profile Picture +############################################################################### + + set_picture: (files, callback) => + input_picture = files[0] + warning_file_format = z.localization.Localizer.get_text z.string.alert_upload_file_format + warning_file_size = z.localization.Localizer.get_text { + id: z.string.alert_upload_too_large + replace: {placeholder: '%no', content: z.config.MAXIMUM_IMAGE_FILE_SIZE / 1024 / 1024} + } + warning_min_size = z.localization.Localizer.get_text z.string.alert_upload_too_small + + if input_picture.size > z.config.MAXIMUM_IMAGE_FILE_SIZE + return @_show_upload_warning warning_file_size, callback + + if not input_picture.type in z.config.SUPPORTED_IMAGE_TYPES + return @_show_upload_warning warning_file_format, callback + + max_width = z.config.MINIMUM_PROFILE_IMAGE_SIZE.WIDTH + max_height = z.config.MINIMUM_PROFILE_IMAGE_SIZE.HEIGHT + z.util.valid_profile_image_size input_picture, max_width, max_height, (valid) => + if valid + @user_repository.change_picture input_picture, callback + else + @_show_upload_warning warning_min_size, callback + + click_on_change_picture: (files) => + @set_picture files, -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.PROFILE_PICTURE_CHANGED, source: 'fromPhotoLibrary' + + _show_upload_warning: (warning, callback) -> + amplify.publish z.event.WebApp.AUDIO.PLAY, z.audio.AudioType.ALERT + setTimeout -> + callback? null, 'error' + window.alert warning + , 200 + +############################################################################### +# Clients +############################################################################### + + # TODO handle clients + on_client_add: (client) => + amplify.publish z.event.WebApp.SEARCH.BADGE.SHOW + @new_clients.push client + + on_show_new_clients: => + amplify.publish z.event.WebApp.SEARCH.BADGE.HIDE + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CONNECTED_DEVICE, + data: @new_clients() + close: => + @new_clients.removeAll() + secondary: => + setTimeout => + @toggle_settings() + , 1000 diff --git a/app/script/view_model/SettingsViewModel.coffee b/app/script/view_model/SettingsViewModel.coffee new file mode 100644 index 00000000000..78a45adc8ea --- /dev/null +++ b/app/script/view_model/SettingsViewModel.coffee @@ -0,0 +1,194 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +SETTING = + ALL: '0' + NONE: '2' + SOME: '1' + +LOCALYTICS_SOUND_SETTING = + ALL: 'alwaysPlay' + SOME: 'FirstMessageOnly' + NONE: 'neverPlay' + +class z.ViewModel.SettingsViewModel + constructor: (@element_id, @user_repository, @conversation_repository, @client_repository, @cryptography_repository) -> + @logger = new z.util.Logger 'z.ViewModel.SelfProfileViewModel', z.config.LOGGER.OPTIONS + + @user = @user_repository.self + + @settings_modal = undefined + + @remove_form_visible = ko.observable false + @remove_form_error = ko.observable false + + @selected_device = ko.observable() + @selected_device.subscribe => + if @selected_device() + @is_resetting_session false + @remove_form_visible false + @remove_form_error false + @_update_fingerprints() + + @fingerprint_remote = ko.observable '' + @fingerprint_local = ko.observable '' + @is_resetting_session = ko.observable false + + @current_client = @client_repository.current_client + # All clients except the current client + @devices = ko.observableArray() + @client_repository.clients.subscribe (client_ets) => + client_ets = client_ets.filter (client_et) => + return client_et.meta.user_id is @user().id and client_et.id isnt @current_client()?.id + @devices client_ets + + @data_setting = ko.observable() + @data_setting.subscribe (setting) => @user_repository.save_property_data_settings setting + + @delete_status = ko.observable 'button' + @delete_confirm_text = ko.observable '' + + @sound_setting = ko.observable() + @sound_setting.subscribe (setting) => + audio_setting = z.audio.AudioSetting.ALL + tracking_value = LOCALYTICS_SOUND_SETTING.ALL + + if setting is SETTING.SOME + audio_setting = z.audio.AudioSetting.SOME + tracking_value = LOCALYTICS_SOUND_SETTING.SOME + else if setting is SETTING.NONE + audio_setting = z.audio.AudioSetting.NONE + tracking_value = LOCALYTICS_SOUND_SETTING.NONE + + @user_repository.save_property_sound_alerts audio_setting + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SOUND_SETTINGS_CHANGED, value: tracking_value + + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATED, @update_properties + amplify.subscribe z.event.WebApp.PROFILE.SETTINGS.SHOW, @toggle_settings + + ko.applyBindings @, document.getElementById @element_id + + + ############################################################################### + # Settings menu + ############################################################################### + + toggle_settings: => + @settings_modal ?= new zeta.webapp.module.Modal '#self-settings', => @selected_device null + if @settings_modal.is_hidden() + @client_repository.get_clients_for_self() + @settings_modal.toggle() + + + ############################################################################### + # Settings + ############################################################################### + + _update_fingerprints: => + @cryptography_repository.get_session @user().id, @selected_device().id + .then (cryptobox_session) => + @fingerprint_remote cryptobox_session.fingerprint_remote() + @fingerprint_local cryptobox_session.fingerprint_local() + + click_on_delete: -> + @delete_confirm_text z.localization.Localizer.get_text + id: z.string.preferences_delete_info + replace: {placeholder: '%email', content: @user().email()} + + @delete_status 'dialog' + + click_on_delete_send: -> + @user_repository.delete_me() + @delete_status 'sent' + setTimeout => + @delete_status 'button' + , 5000 + + click_on_delete_cancel: -> + @delete_status 'button' + + click_on_reset_password: -> + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.PASSWORD_RESET, value: 'fromProfile' + (window.open z.string.url_password_reset)?.focus() + + click_on_device: (client_et) => + @selected_device client_et + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SETTINGS.VIEWED_DEVICE, outcome: 'success' + + click_on_device_close: => + @selected_device null + + click_on_verify_client: => + toggle_verified = !!!@selected_device().meta.is_verified() + client_id = @selected_device().id + user_id = @user().id + changes = + meta: + is_verified: toggle_verified + + @client_repository.update_client_in_db user_id, client_id, changes + .then => @selected_device().meta.is_verified toggle_verified + + click_on_reset_session: => + reset_progress = => + window.setTimeout => + @is_resetting_session false + , 550 + + @is_resetting_session true + @conversation_repository.reset_session @user().id, @selected_device().id, @conversation_repository.self_conversation().id + .then -> reset_progress() + .catch -> reset_progress() + + click_on_remove_device_submit: (password) => + @client_repository.delete_client @selected_device().id, password + .then => + @selected_device null + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SETTINGS.REMOVED_DEVICE, outcome: 'success' + .catch => + @logger.log @logger.levels.WARN, 'Unable to remove device' + @remove_form_error true + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.SETTINGS.REMOVED_DEVICE, outcome: 'fail' + + update_properties: (properties) => + if properties.settings.sound.alerts is z.audio.AudioSetting.ALL + @sound_setting SETTING.ALL + else if properties.settings.sound.alerts is z.audio.AudioSetting.SOME + @sound_setting SETTING.SOME + else if properties.settings.sound.alerts is z.audio.AudioSetting.NONE + @sound_setting SETTING.NONE + + @data_setting properties.settings.privacy.report_errors + + + ############################################################################### + # Google/Contacts upload + ############################################################################### + + connect_google: -> + amplify.publish z.event.WebApp.CONNECT.IMPORT_CONTACTS, z.connect.ConnectSource.GMAIL, z.connect.ConnectTrigger.SETTINGS + amplify.publish z.event.WebApp.SEARCH.SHOW + @settings_modal.hide() + + connect_osx_contacts: -> + amplify.publish z.event.WebApp.CONNECT.IMPORT_CONTACTS, z.connect.ConnectSource.ICLOUD, z.connect.ConnectTrigger.SETTINGS + amplify.publish z.event.WebApp.SEARCH.SHOW + @settings_modal.hide() diff --git a/app/script/view_model/StartUIViewModel.coffee b/app/script/view_model/StartUIViewModel.coffee new file mode 100644 index 00000000000..4fe5667cdde --- /dev/null +++ b/app/script/view_model/StartUIViewModel.coffee @@ -0,0 +1,486 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +class z.ViewModel.StartUIViewModel + constructor: (element_id, @conversation_repository, @search_repository, @user_repository, @connect_repository) -> + @logger = new z.util.Logger 'z.ViewModel.StartUIViewModel', z.config.LOGGER.OPTIONS + + @search = _.debounce (query) => + query = query.trim() + if query + @clear_search_results() + + # contacts + @search_results.contacts @user_repository.get_user_by_name query + + # search for groups + @search_results.groups @conversation_repository.get_groups_by_name query + + # search for others + @search_repository.search_by_name query + .then (user_ets) => + if query is @search_input().trim() + @search_results.others user_ets + else + @logger.log @logger.levels.INFO, "Resolved Search query #{query} is outdated" + .catch (error) => + @logger.log @logger.levels.ERROR, "Error searching for contacts: #{error.message}", error + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.BOOLEAN.SEARCHED_FOR_PEOPLE, true + , 300 + + @user = @user_repository.self + + @search_input = ko.observable '' + @search_input.subscribe @search + @selected_people = ko.observableArray [] + + @has_created_conversation = ko.observable false + @show_hint = ko.computed => @selected_people().length is 1 and not @has_created_conversation() + + @group_hint_text = z.localization.Localizer.get_text z.string.search_group_hint + + # results + @top_users = ko.observableArray [] + @suggestions = ko.observableArray [] + @connections = ko.computed => + @conversation_repository.sorted_conversations() + .filter (conversation_et) -> conversation_et.type() is z.conversation.ConversationType.ONE2ONE + @connections.extend rateLimit: 500 + + @search_results = + groups: ko.observableArray [] + contacts: ko.observableArray [] + others: ko.observableArray [] + + # view states + @show_no_contacts_on_wire = ko.observable false + @is_searching = ko.observable false + @show_spinner = ko.observable false + @show_people_picker = ko.observable() + + @has_uploaded_contacts = ko.observable false + + @has_results = ko.computed => + return @search_results.groups().length > 0 or + @search_results.contacts().length > 0 or + @search_results.others().length > 0 + + @show_top_people = ko.computed => + return !!@top_users().length + + @show_suggestions = ko.computed => + return !!@suggestions().length + + @show_connections = ko.computed => + return @top_users().length > 9 and not @show_suggestions() + + @show_search_results = ko.computed => + if @selected_people().length is 0 and @search_input().length is 0 + @clear_search_results() + return false + return @has_results() or @search_input().length > 0 + + @show_invite = ko.computed => + no_top_people_and_suggestions = not @show_search_results() and not @show_top_people() and not @show_suggestions() + no_search_results = @show_search_results() and not @has_results() and not @is_searching() + return no_top_people_and_suggestions or no_search_results + + # invite bubble states + @show_invite_form = ko.observable true + @show_invite_form_only = ko.computed => + return true if @has_uploaded_contacts() + return true if not @has_uploaded_contacts() and not @show_top_people() and not @show_suggestions() + return false + + # selected user + @user_profile = ko.observable null + @user_bubble = null + + # invite bubble + @invite_bubble = null + @invite_message = ko.observable '' + @invite_message_selected = ko.observable true + @invite_hints = ko.computed => + meta_key_mac = z.localization.Localizer.get_text z.string.invite_meta_key_mac + meta_key_pc = z.localization.Localizer.get_text z.string.invite_meta_key_pc + meta_key = if z.util.Environment.os.mac then meta_key_mac else meta_key_pc + + if @invite_message_selected() + return z.localization.Localizer.get_text + id: z.string.invite_hint_selected + replace: [ + placeholder: '%meta_key', content: meta_key + ] + else + return z.localization.Localizer.get_text + id: z.string.invite_hint_unselected + replace: [ + placeholder: '%meta_key', content: meta_key + ] + + @invite_button_text = ko.computed => + button_text = if @show_invite_form_only() then z.string.people_invite else z.string.people_bring_your_friends + z.localization.Localizer.get_text button_text + + # last open bubble + @user_bubble = undefined + @user_bubble_last_id = undefined + + amplify.subscribe z.event.WebApp.CONNECT.IMPORT_CONTACTS, @import_contacts + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATED, @update_properties + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATE.GOOGLE, @update_properties + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATE.OSX_CONTACTS, @update_properties + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATE.HAS_CREATED_CONVERSATION, @update_properties + amplify.subscribe z.event.WebApp.PROPERTIES.UPDATED, @update_properties + amplify.subscribe z.event.WebApp.SEARCH.HIDE, @close + amplify.subscribe z.event.WebApp.SEARCH.SHOW, @open + + ko.applyBindings @, document.getElementById element_id + + clear_search_results: -> + @search_results.groups.removeAll() + @search_results.contacts.removeAll() + @search_results.others.removeAll() + @is_searching false + @show_no_contacts_on_wire false + + click_on_close: => + @_close() + + _track_import: (source, ui_identifier, error) -> + if ui_identifier is z.connect.ConnectTrigger.ONBOARDING + event_name = z.tracking.EventName.ONBOARDING.IMPORTED_CONTACTS + else + event_name = z.tracking.EventName.SETTINGS.IMPORTED_CONTACTS + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, event_name, + source: source + outcome: if error then 'fail' else 'success' + + ### + Connect with contacts. + @param source [z.connect.ConnectSource] Source for the contacts import + @param ui_identifier [String] UI component that triggered the event (used for Analytics) + ### + import_contacts: (source, ui_identifier = z.connect.ConnectTrigger.ONBOARDING) => + @show_spinner true + if source is z.connect.ConnectSource.GMAIL + import_promise = @connect_repository.get_google_contacts() + else if source is z.connect.ConnectSource.ICLOUD + import_promise = @connect_repository.get_osx_contacts() + + import_promise.then (response) => + @_show_onboarding_results response + .catch (error) => + if error.type isnt z.connect.ConnectError::TYPE.NO_CONTACTS + @logger.log @logger.levels.ERROR, "Importing contacts from '#{source}' failed: #{error.message}", error + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CONTACTS, action: => + @import_contacts source, ui_identifier + .then (error) => + @show_spinner false + @_track_import source, ui_identifier, error + + _show_onboarding_results: (response) => + @search_repository.show_onboarding response + .then (matched_user_ets) => + @suggestions matched_user_ets + return @search_repository.get_top_people() + .then (user_ets) => + @top_users user_ets + @selected_people.removeAll() + if @suggestions().length is 0 + if @top_users().length > 0 + @suggestions @top_users() + else + @show_no_contacts_on_wire true + .catch (error) => + @logger.log @logger.levels.ERROR, "Could not show the on-boarding results: #{error.message}", error + + open: (update = true) => + $(document).on 'keydown.show_search', (event) => @_close() if event.keyCode is z.util.KEYCODE.ESC + + if update + @search_repository.get_top_people() + .then (user_ets) => + @top_users user_ets if user_ets.length > 0 + .catch (error) => + @logger.log @logger.levels.ERROR, "Could not update the top people: #{error.message}", error + + @show_spinner false + + # clean up + @suggestions.removeAll() + @selected_people.removeAll() + @clear_search_results() + @user_profile null + @_show() + + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.SessionEventName.INTEGER.SEARCH_OPENED + + _close: -> + amplify.publish z.event.WebApp.SEARCH.HIDE + + close: => + $(document).off 'keydown.show_search' + + @user_bubble?.hide() + @invite_bubble?.hide() + @show_spinner false + + @selected_people.removeAll() + @search_input '' + @_hide() + + amplify.publish z.event.WebApp.CONVERSATION_LIST.SHOW + amplify.publish z.event.WebApp.LIST.BLUR, 0 + + click_on_group: (conversation_et) => + @conversation_repository.unarchive_conversation conversation_et if conversation_et.is_archived() + @_close() + amplify.publish z.event.WebApp.CONVERSATION.SHOW, conversation_et + + click_on_other: (user_et, e) => + + create_bubble = (element_id) => + @user_profile user_et + @user_bubble_last_id = element_id + @user_bubble = new zeta.webapp.module.Bubble + host_selector: "##{element.attr('id')}" + scroll_selector: '.start-ui-list' + on_hide: => + @user_bubble = undefined + @user_bubble_last_id = undefined + on_show: -> + $('.start-ui-user-bubble .user-profile-connect-message').focus() + + @user_bubble.toggle() + + # we clicked on the same bubble + if @user_bubble and @user_bubble_last_id is e.currentTarget.id + @user_bubble.toggle() + return + + element = $(e.currentTarget).attr + 'id': Date.now() + 'data-bubble': '#start-ui-user-bubble' + 'data-placement': 'right-flex' + + # dismiss old bubble and wait with creating the new one when another bubble is open + if @user_bubble + @user_bubble?.hide() + window.setTimeout -> + create_bubble(element[0].id) + , 550 + else + create_bubble(element[0].id) + + _collapse_item: (search_list_item, callback) -> + search_list_item.find('.search-list-item-connect').remove() + window.requestAnimFrame -> + search_list_item + .addClass 'search-list-item-collapse' + .on z.util.alias.animationend, (event) -> + if event.originalEvent.propertyName is 'height' + search_list_item + .remove() + .off z.util.alias.animationend + callback?() + + + click_on_dismiss: (user_et, event) => + search_list_item = $(event.currentTarget.parentElement.parentElement) + @_collapse_item search_list_item, => + @search_repository.ignore_suggestion user_et.id + .then => + @suggestions.remove user_et + .catch (error) => + @logger.log @logger.levels.ERROR, "Failed to ignore suggestions: '#{error.message}'", error + + click_on_connect: (user_et, event) => + search_list_item = $(event.currentTarget.parentElement.parentElement) + search_list_item + .addClass 'search-list-item-connect-anim' + .one z.util.alias.animationend, => + window.setTimeout => + @_collapse_item search_list_item, => + @user_repository.create_connection user_et + .then => + @suggestions.remove user_et + , 550 + + + ############################################################################### + # USER BUBBLE + ############################################################################### + + on_user_accept: (user_et) => + @_close() + @user_repository.accept_connection_request user_et, true + + on_user_connect: => + @_close() + + on_user_ignore: (user_et) => + @user_repository.ignore_connection_request user_et + .then => + @user_bubble?.hide() + + on_user_open: => + @_close() + + on_user_unblock: (user_et) => + @_close() + @user_repository.unblock_user user_et, true + + on_cancel_request: => + @user_bubble?.hide() + + + ############################################################################### + # INVITE BUBBLE + ############################################################################### + + click_on_contacts_import: => + @invite_bubble?.hide() + @import_contacts z.connect.ConnectSource.ICLOUD, z.connect.ConnectTrigger.SEARCH + + click_on_gmail_import: => + @invite_bubble?.hide() + @import_contacts z.connect.ConnectSource.GMAIL, z.connect.ConnectTrigger.SEARCH + + click_on_import_form: => + @show_invite_form false + + click_on_invite_form: => + @show_invite_form true + @_focus_invite_form() + + show_invite_bubble: => + return if @invite_bubble? + + self = @user_repository.self() + + @invite_message z.localization.Localizer.get_text + id: z.string.invite_message + replace: [ + {placeholder: '%url', content: z.util.Invite.get_invitation_to_connect_url self.id} + ] + + @invite_bubble = new zeta.webapp.module.Bubble + host_selector: '#invite-button' + scroll_selector: '.start-ui-list' + on_hide: => + $('.invite-link-box .bg').removeClass 'bg-animation' + $('.invite-link-box .message').off 'copy blur focus' + @invite_bubble = null + @show_invite_form true + on_show: => @_focus_invite_form() if @show_invite_form() + + @invite_bubble.show() + + _focus_invite_form: => + $('.invite-link-box .message') + .on 'copy', (e) => + $(e.currentTarget).parent().find('.bg') + .addClass 'bg-animation' + .on z.util.alias.animationend, (e) => + return if e.originalEvent.animationName isnt 'message-bg-fadeout' + $(@).off z.util.alias.animationend + @invite_bubble.hide() + .on 'blur', => + @invite_message_selected false + .on 'click', (e) => + @invite_message_selected true + $(e.target).select() + .trigger 'click' + + + ############################################################################### + # USER PROPERTIES + ############################################################################### + + update_properties: => + @has_created_conversation @user_repository.properties.has_created_conversation + @has_uploaded_contacts @user_repository.properties.contact_import.google? or @user_repository.properties.contact_import.osx? + return true + + ############################################################################### + # H + ############################################################################### + + on_submit_search: (callback) => + user_ids = (user_et.id for user_et in @selected_people()) + + if user_ids.length is 0 + return + + if user_ids.length is 1 + conversation_et = @conversation_repository.get_one_to_one_conversation user_ids[0] + @click_on_group conversation_et + callback conversation_et if _.isFunction callback + return + + on_success = (conversation_et) => + @logger.log @logger.levels.INFO, "Created new conversation with ID: #{conversation_et.id}" + @user_repository.save_property_has_created_conversation() + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CONVERSATION.CREATE_GROUP_CONVERSATION, + {creationContext: 'search', numberOfParticipants: user_ids.length} + @click_on_group conversation_et + callback conversation_et if _.isFunction callback + + # Error: {code: 403, message: "Users are not connected", label: "not-connected"} + on_error = (error) => + @logger.log @logger.levels.WARN, "Unable to create conversation: #{error.message}" + @_close() + + @conversation_repository.create_new_conversation user_ids, null, on_success, on_error + + on_audio_call: => + @on_submit_search (conversation_et) -> + window.setTimeout -> + amplify.publish z.event.WebApp.CALL.STATE.JOIN, conversation_et.id + , 1000 + + on_video_call: => + @on_submit_search (conversation_et) -> + window.setTimeout -> + amplify.publish z.event.WebApp.CALL.STATE.JOIN, conversation_et.id, true + , 1000 + + on_photo: (images) => + @on_submit_search -> + window.setTimeout -> + amplify.publish z.event.WebApp.CONVERSATION.IMAGE.SEND, images + , 1000 + +############################################################################### +# Start UI animations +############################################################################### + _hide: -> + $('#start-ui').removeClass 'start-ui-is-visible' + $('user-input input').blur() + + _show: -> + $('#start-ui').addClass('start-ui-is-visible').find('.start-ui-list').scrollTop 0 + $('user-input input').focus() + @show_people_picker Date.now() diff --git a/app/script/view_model/VideoCallingViewModel.coffee b/app/script/view_model/VideoCallingViewModel.coffee new file mode 100644 index 00000000000..7bc66088724 --- /dev/null +++ b/app/script/view_model/VideoCallingViewModel.coffee @@ -0,0 +1,212 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + + +class z.ViewModel.VideoCallingViewModel + constructor: (element_id, @call_center, @user_repository, @conversation_repository) -> + @logger = new z.util.Logger 'z.ViewModel.VideoCallingViewModel', z.config.LOGGER.OPTIONS + + @self_user = @user_repository.self + + @available_devices = @call_center.media_devices_handler.available_devices + @current_device_id = @call_center.media_devices_handler.current_device_id + @current_device_index = @call_center.media_devices_handler.current_device_index + + @local_video_stream = @call_center.media_stream_handler.local_media_streams.video + @remote_video_stream = @call_center.media_stream_handler.remote_media_streams.video + + @self_stream_state = @call_center.media_stream_handler.self_stream_state + + @is_choosing_screen = ko.observable false + @is_minimized = ko.observable false + + @minimize_timeout = undefined + + @number_of_screen_devices = ko.observable 0 + @number_of_video_devices = ko.observable 0 + @video_mode = ko.observable z.calling.enum.VideoOrientation.LANDSCAPE + + @joined_call = @call_center.joined_call + + @videod_call = ko.pureComputed => + for call_et in @call_center.calls() + is_active = call_et.state() in z.calling.enum.CallStateGroups.IS_ACTIVE + is_client_videod = call_et.self_client_joined() and @self_stream_state.screen_shared() or @self_stream_state.videod() + is_remote_videod = call_et.is_remote_videod() and not call_et.is_ongoing_on_another_client() + return call_et if is_active and (is_client_videod or is_remote_videod or @is_choosing_screen()) + + @is_ongoing = ko.pureComputed => + return @videod_call()? and @joined_call()?.state() is z.calling.enum.CallState.ONGOING + + @overlay_icon_class = ko.pureComputed => + if @is_ongoing() + if @self_stream_state.muted() + return 'icon-mute' + else if not @self_stream_state.screen_shared() and not @self_stream_state.videod() + return 'icon-video-off' + + @remote_user = ko.pureComputed => + return @joined_call()?.participants()[0]?.user + + @show_local = ko.pureComputed => + return not @is_minimized() and not @is_choosing_screen() + @show_local_video = ko.pureComputed => + is_visible = @self_stream_state.screen_shared() or @self_stream_state.videod() or @videod_call()?.state() isnt z.calling.enum.CallState.ONGOING + return @local_video_stream() and is_visible + + @show_remote = ko.pureComputed => + return @show_remote_video() or @show_remote_participant() or @is_choosing_screen() + @show_remote_participant = ko.pureComputed => + is_visible = @remote_user() and not @is_minimized() and not @joined_call()?.is_remote_videod() or not @remote_video_stream() + return @is_ongoing() and is_visible + @show_remote_video = ko.pureComputed => + is_visible = @joined_call()?.is_remote_videod() and @remote_video_stream() + return @is_ongoing() and is_visible + + @show_switch_camera = ko.pureComputed => + is_visible = @local_video_stream() and @available_devices.video_input().length > 1 and @self_stream_state.videod() + return @is_ongoing() and is_visible + @show_switch_screen = ko.pureComputed => + is_visible = @local_video_stream() and @available_devices.screen_input().length > 1 and @self_stream_state.screen_shared() + return @is_ongoing() and is_visible + + @show_controls = ko.pureComputed => + is_visible = @show_remote_video() or @show_remote_participant() and not @is_minimized() + return @is_ongoing() and is_visible + @show_toggle_screen = ko.pureComputed -> + return z.calling.CallCenter.supports_screen_sharing() + @disable_toggle_screen = ko.pureComputed => + return @joined_call()?.is_remote_screen_shared() + + @videod_call.subscribe (videod_call) => + if videod_call + if @show_local_video() or @show_remote_video() or @is_choosing_screen() or videod_call.state() is z.calling.enum.CallState.INCOMING + @is_minimized false + @logger.log @logger.levels.INFO, "Displaying call '#{videod_call.id}' full-screen", videod_call + else + @is_minimized true + @logger.log @logger.levels.INFO, "Minimizing call '#{videod_call.id}' that is not videod", videod_call + else + @is_minimized false + @logger.log @logger.levels.INFO, 'Resetting full-screen calling to maximize' + + @available_devices.screen_input.subscribe (media_devices) => + if _.isArray media_devices then @number_of_screen_devices media_devices.length else @number_of_screen_devices 0 + @available_devices.video_input.subscribe (media_devices) => + if _.isArray media_devices then @number_of_video_devices media_devices.length else @number_of_video_devices 0 + @show_remote_participant.subscribe (show_remote_participant) => + if @minimize_timeout + window.clearTimeout @minimize_timeout + @minimize_timeout = undefined + + if show_remote_participant and @videod_call() and not @is_choosing_screen() + @logger.log @logger.levels.INFO, "Scheduled minimizing call '#{@videod_call().id}' on timeout as remote user '#{@remote_user()?.name()}' is not videod" + @minimize_timeout = window.setTimeout => + @is_minimized true if not @is_choosing_screen() + @logger.log @logger.levels.INFO, "Minimizing call '#{@videod_call().id}' on timeout as remote user '#{@remote_user()?.name()}' is not videod" + , 5000 + + amplify.subscribe z.event.WebApp.CALL.STATE.TOGGLE_SCREEN, @choose_shared_screen + + ko.applyBindings @, document.getElementById element_id + + choose_shared_screen: (conversation_id) => + return if @disable_toggle_screen() + + if @self_stream_state.screen_shared() or z.util.Environment.browser.firefox + @call_center.state_handler.toggle_screen conversation_id + + else if z.util.Environment.electron + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CALLING.SHARED_SCREEN, + conversation_type: if @joined_call().is_group() then 'group' else 'one_to_one' + kind_of_call_when_sharing: if @joined_call().is_remote_videod() then 'video' else 'audio' + + @call_center.media_devices_handler.get_screen_sources() + .then (screen_sources) => + if screen_sources.length > 1 + @is_minimized false + @is_choosing_screen true + else + @call_center.state_handler.toggle_screen conversation_id + .catch (error) => + @logger.log @logger.levels.ERROR, 'Unable to get screens sources for sharing', error + + clicked_on_cancel_call: => + @call_center.state_handler.leave_call @joined_call()?.id + + clicked_on_cancel_screen: => + @is_choosing_screen false + + clicked_on_mute_audio: => + @call_center.state_handler.toggle_audio @joined_call()?.id + + clicked_on_share_screen: => + @choose_shared_screen @joined_call().id + + clicked_on_choose_screen: (screen_source) => + @current_device_id.screen_input '' + @logger.log @logger.levels.INFO, "Selected '#{screen_source.name}' for screen sharing", screen_source + @is_choosing_screen false + @current_device_id.screen_input screen_source.id + @call_center.state_handler.toggle_screen @joined_call().id + if not @joined_call().is_remote_videod() + @is_minimized true + @logger.log @logger.levels.INFO, "Minimizing call '#{@videod_call().id}' on screen selection as remote user '#{@remote_user()?.name()}' is not videod" + + clicked_on_stop_video: => + @call_center.state_handler.toggle_video @joined_call()?.id + + clicked_on_toggle_camera: => + @call_center.media_devices_handler.toggle_next_camera() + + clicked_on_toggle_screen: => + @call_center.media_devices_handler.toggle_next_screen() + + clicked_on_minimize: => + @is_minimized true + @logger.log @logger.levels.INFO, "Minimizing call '#{@videod_call().id}' on user click" + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.CALLING.MINIMIZED_FROM_FULLSCREEN, + conversation_type: if @joined_call().is_group() then 'group' else 'one_to_one' + + clicked_on_maximize: => + @is_minimized false + @logger.log @logger.levels.INFO, "Maximizing call '#{@videod_call().id}' on user click" + + # Detect the aspect ratio of a MediaElement and set the video mode. + on_loadedmetadata: (view_model, event) => + media_element = event.target + if media_element.videoHeight > media_element.videoWidth + detected_video_mode = z.calling.enum.VideoOrientation.PORTRAIT + else + detected_video_mode = z.calling.enum.VideoOrientation.LANDSCAPE + @video_mode detected_video_mode + @logger.log @logger.levels.INFO, "Video is in '#{detected_video_mode}' mode" + + +# http://stackoverflow.com/questions/28762211/unable-to-mute-html5-video-tag-in-firefox +ko.bindingHandlers.mute_media_element = + update: (element, valueAccessor) -> + element.muted = true if valueAccessor() + + +ko.bindingHandlers.source_stream = + update: (element, valueAccessor) -> + element.srcObject = valueAccessor() diff --git a/app/script/view_model/WarningsViewModel.coffee b/app/script/view_model/WarningsViewModel.coffee new file mode 100644 index 00000000000..000a11cbe44 --- /dev/null +++ b/app/script/view_model/WarningsViewModel.coffee @@ -0,0 +1,110 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +# Types for warning banners +z.ViewModel.WarningType = + # Permission requests: dimmed screen, warning bar + REQUEST_CAMERA: 'request_camera' + REQUEST_MICROPHONE: 'request_microphone' + REQUEST_NOTIFICATION: 'request_notification' + REQUEST_SCREEN: 'request_screen' +# Permission callbacks: !dimmed screen, warning bar + DENIED_CAMERA: 'camera_access_denied' + DENIED_MICROPHONE: 'mic_access_denied' + DENIED_SCREEN: 'screen_access_denied' + NOT_FOUND_CAMERA: 'not_found_camera' + NOT_FOUND_MICROPHONE: 'not_found_microphone' + UNSUPPORTED_INCOMING_CALL: 'unsupported_incoming_call' + UNSUPPORTED_OUTGOING_CALL: 'unsupported_outgoing_call' + CONNECTIVITY_RECONNECT: 'connectivity_reconnect' + CONNECTIVITY_RECOVERY: 'connectivity_recovery' + NO_INTERNET: 'no_internet' + +class z.ViewModel.WarningsViewModel + constructor: (element_id) -> + @logger = new z.util.Logger 'z.ViewModel.WarningsViewModel', z.config.LOGGER.OPTIONS + + # Array of warning banners + @warnings = ko.observableArray() + @top_warning = ko.computed => + return @warnings()[@warnings().length - 1] + , @, deferEvaluation: true + @warnings.subscribe (warnings) -> + mini_modes = [ + z.ViewModel.WarningType.CONNECTIVITY_RECONNECT + z.ViewModel.WarningType.NO_INTERNET + ] + if warnings.length is 0 + top_margin = '0' + else if warnings[warnings.length - 1] is z.ViewModel.WarningType.CONNECTIVITY_RECOVERY + top_margin = '0' + else if warnings[warnings.length - 1] in mini_modes + top_margin = '32px' + else + top_margin = '64px' + $('#app').css top: top_margin + requestAnimFrame -> $(window).trigger 'resize' + + @first_name = ko.observable() + @call_id = undefined + + @warning_dimmed = ko.computed => + for warning in @warnings() + return true if warning in [ + z.ViewModel.WarningType.REQUEST_CAMERA + z.ViewModel.WarningType.REQUEST_MICROPHONE + z.ViewModel.WarningType.REQUEST_NOTIFICATION + z.ViewModel.WarningType.REQUEST_SCREEN + ] + return false + , @, deferEvaluation: true + @warning_dimmed.extend rateLimit: 200 + + amplify.subscribe z.event.WebApp.WARNINGS.SHOW, @show_warning + amplify.subscribe z.event.WebApp.WARNINGS.DISMISS, @dismiss_warning + + ko.applyBindings @, document.getElementById element_id + + # Function is used to close a warning banner by clicking on it's close (X) button + close_warning: => + warning_to_remove = @top_warning() + @dismiss_warning warning_to_remove + + switch warning_to_remove + when z.ViewModel.WarningType.REQUEST_MICROPHONE + amplify.publish z.event.WebApp.WARNINGS.MODAL, z.ViewModel.ModalType.CALLING, + action: -> (window.open z.localization.Localizer.get_text z.string.url_support_mic_access_denied)?.focus() + when z.ViewModel.WarningType.REQUEST_NOTIFICATION + # We block subsequent permission requests for notifications when the user ignores the request. + amplify.publish z.event.WebApp.SYSTEM_NOTIFICATION.REQUEST_PERMISSION, false + + dismiss_warning: (type) => + type = @top_warning() if not type + @logger.log @logger.levels.WARN, "Dismissed warning of type '#{type}'" + @warnings.remove type + + show_warning: (type, info) => + @dismiss_warning() if @top_warning() and type in [z.ViewModel.WarningType.CONNECTIVITY_RECONNECT, z.ViewModel.WarningType.NO_INTERNET] + @logger.log @logger.levels.WARN, "Showing warning of type '#{type}'" + if info? + @first_name info.first_name + @call_id = info.call_id + @warnings.push type diff --git a/app/script/view_model/WelcomeViewModel.coffee b/app/script/view_model/WelcomeViewModel.coffee new file mode 100644 index 00000000000..1ad9d06da45 --- /dev/null +++ b/app/script/view_model/WelcomeViewModel.coffee @@ -0,0 +1,63 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.WelcomeViewModel + constructor: (element_id, @user_repository) -> + @logger = new z.util.Logger 'z.ViewModel.WelcomeViewModel', z.config.LOGGER.OPTIONS + + @user = @user_repository.self + + @show_welcome = ko.observable false + @uploading_image = ko.observable false + @unsplash_image_loaded = ko.observable false + @disable_keep_button = ko.computed => + @uploading_image() or not @unsplash_image_loaded() + + ko.applyBindings @, document.getElementById element_id + + amplify.subscribe z.event.WebApp.WELCOME.SHOW, => @show_welcome true + amplify.subscribe z.event.WebApp.WELCOME.UNSPLASH_LOADED, => @unsplash_image_loaded true + + close_welcome: => + @show_welcome false + amplify.publish z.event.WebApp.APP.FADE_IN + + if @user_repository.connections().length is 0 + setTimeout -> + amplify.publish z.event.WebApp.SEARCH.SHOW + , 550 + + choose_your_own_picture: (files) => + return if @uploading_image() + + @uploading_image true + amplify.publish z.event.WebApp.PROFILE.UPLOAD_PICTURE, files, (response, error) => + @uploading_image false + @close_welcome() if not error + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ONBOARDING.ADDED_PHOTO, + source: 'gallery' + outcome: if error then 'fail' else 'success' + + keep_this_picture: => + amplify.publish z.event.WebApp.ANALYTICS.EVENT, z.tracking.EventName.ONBOARDING.ADDED_PHOTO, + source: 'unsplash' + outcome: 'success' + @close_welcome() diff --git a/app/script/view_model/WindowTitleViewModel.coffee b/app/script/view_model/WindowTitleViewModel.coffee new file mode 100644 index 00000000000..8709462d0d3 --- /dev/null +++ b/app/script/view_model/WindowTitleViewModel.coffee @@ -0,0 +1,68 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +window.z ?= {} +z.ViewModel ?= {} + +class z.ViewModel.WindowTitleViewModel + constructor: (@content, @user_repository, @conversation_repository) -> + @logger = new z.util.Logger 'z.ViewModel.WindowTitleViewModel', z.config.LOGGER.OPTIONS + amplify.subscribe z.event.WebApp.LOADED, @initiate_title_updates + + initiate_title_updates: => + @logger.log @logger.levels.INFO, 'Starting to update window title' + ko.computed => + + window_title = '' + badge_count = 0 + number_of_unread_conversations = 0 + number_of_connect_requests = @user_repository.connect_requests().length + + @conversation_repository.conversations_unarchived().forEach (conversation_et) -> + if conversation_et.type() isnt z.conversation.ConversationType.CONNECT + number_of_unread_conversations++ if conversation_et.number_of_unread_messages() > 0 + + badge_count = number_of_connect_requests + number_of_unread_conversations + + if badge_count > 0 + window_title += "(#{badge_count}) " + + amplify.publish z.event.WebApp.CONVERSATION.UNREAD, badge_count + + switch @content.state() + when z.ViewModel.CONTENT_STATE.PENDING + if number_of_connect_requests > 1 + window_title += z.localization.Localizer.get_text { + id: z.string.conversation_list_many_connection_request + replace: {placeholder: '%no', content: number_of_connect_requests} + } + else + window_title += z.localization.Localizer.get_text z.string.conversation_list_one_connection_request + when z.ViewModel.CONTENT_STATE.CONVERSATION + window_title += @conversation_repository.active_conversation()?.display_name() + when z.ViewModel.CONTENT_STATE.PROFILE + window_title += @user_repository.self().name() + + if window_title is '' + window_title = z.localization.Localizer.get_text z.string.wire + else + window_title += " - #{z.localization.Localizer.get_text z.string.wire}" + + window.document.title = window_title + + .extend rateLimit: 750 diff --git a/app/script/view_model/bindings/CommonBindings.coffee b/app/script/view_model/bindings/CommonBindings.coffee new file mode 100644 index 00000000000..c0539b16c46 --- /dev/null +++ b/app/script/view_model/bindings/CommonBindings.coffee @@ -0,0 +1,287 @@ +# +# Wire +# Copyright (C) 2016 Wire Swiss GmbH +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# + +# use it on the drop area +ko.bindingHandlers.drop_file = + init: (element, valueAccessor, allBindings, data, context) -> + fileDragOver = (data, e) -> + e.preventDefault() + e.currentTarget.classList.add 'drag-hover' + + fileDragLeave = (data, e) -> + e.currentTarget.classList.remove 'drag-hover' + + fileSelectHandler = (data, e) -> + e.preventDefault() + e.currentTarget.classList.remove 'drag-hover' + files = e.dataTransfer?.files or e.originalEvent?.dataTransfer.files + valueAccessor().call @, files if files.length > 0 + + ko.applyBindingsToNode element, + event: + dragover: fileDragOver + dragleave: fileDragLeave + drop: fileSelectHandler + , context + +# blockes the default behaviour when dropping a file on the element +# if an element inside that element is listening to drag events, than this will be triggered after +ko.bindingHandlers.ignore_drop_file = + init: (element, valueAccessor, allBindings, data, context) -> + ko.applyBindingsToNode element, + event: + dragover: (data, e) -> e.preventDefault() + drop: (data, e) -> e.preventDefault() + , context + + +# indicate that the current binding loop should not try to bind this element’s children +# http://www.knockmeout.net/2012/05/quick-tip-skip-binding.html +ko.bindingHandlers.stopBinding = + init: -> + controlsDescendantBindings: true + +ko.virtualElements.allowedBindings.stopBinding = true + +# resize textarea according to the containing text +# Link: http://jsfiddle.net/C8e4w/1/ +# Docu: http://knockoutjs.com/documentation/custom-bindings.html +# +# Example: HTML binding +#