1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.server.notification; 18 19 import static android.app.NotificationManager.IMPORTANCE_HIGH; 20 21 import android.annotation.NonNull; 22 import android.app.NotificationChannel; 23 import android.app.NotificationChannelGroup; 24 import android.stats.sysui.NotificationEnums; 25 26 import com.android.internal.logging.UiEvent; 27 import com.android.internal.logging.UiEventLogger; 28 import com.android.internal.util.FrameworkStatsLog; 29 30 /** 31 * Interface for logging NotificationChannelModified statsd atoms. Provided as an interface to 32 * enable unit-testing - use standard implementation NotificationChannelLoggerImpl in production. 33 */ 34 public interface NotificationChannelLogger { 35 // The logging interface. Not anticipating a need to override these high-level methods, which by 36 // default forward to a lower-level interface. 37 38 /** 39 * Log the creation of a notification channel. 40 * @param channel The channel. 41 * @param uid UID of app that owns the channel. 42 * @param pkg Package of app that owns the channel. 43 */ logNotificationChannelCreated(@onNull NotificationChannel channel, int uid, String pkg)44 default void logNotificationChannelCreated(@NonNull NotificationChannel channel, int uid, 45 String pkg) { 46 logNotificationChannel( 47 NotificationChannelEvent.getCreated(channel), 48 channel, uid, pkg, 0, getLoggingImportance(channel)); 49 } 50 51 /** 52 * Log the deletion of a notification channel. 53 * @param channel The channel. 54 * @param uid UID of app that owns the channel. 55 * @param pkg Package of app that owns the channel. 56 */ logNotificationChannelDeleted(@onNull NotificationChannel channel, int uid, String pkg)57 default void logNotificationChannelDeleted(@NonNull NotificationChannel channel, int uid, 58 String pkg) { 59 logNotificationChannel( 60 NotificationChannelEvent.getDeleted(channel), 61 channel, uid, pkg, getLoggingImportance(channel), 0); 62 } 63 64 /** 65 * Log the modification of a notification channel. 66 * @param channel The channel. 67 * @param uid UID of app that owns the channel. 68 * @param pkg Package of app that owns the channel. 69 * @param oldLoggingImportance Previous logging importance level of the channel. 70 * @param byUser True if the modification was user-specified. 71 */ logNotificationChannelModified(@onNull NotificationChannel channel, int uid, String pkg, int oldLoggingImportance, boolean byUser)72 default void logNotificationChannelModified(@NonNull NotificationChannel channel, int uid, 73 String pkg, int oldLoggingImportance, boolean byUser) { 74 logNotificationChannel(NotificationChannelEvent.getUpdated(byUser), 75 channel, uid, pkg, oldLoggingImportance, getLoggingImportance(channel)); 76 } 77 78 /** 79 * Log the creation or modification of a notification channel group. 80 * @param channelGroup The notification channel group. 81 * @param uid UID of app that owns the channel. 82 * @param pkg Package of app that owns the channel. 83 * @param isNew True if this is a creation of a new group. 84 * @param wasBlocked 85 */ logNotificationChannelGroup(@onNull NotificationChannelGroup channelGroup, int uid, String pkg, boolean isNew, boolean wasBlocked)86 default void logNotificationChannelGroup(@NonNull NotificationChannelGroup channelGroup, 87 int uid, String pkg, boolean isNew, boolean wasBlocked) { 88 logNotificationChannelGroup(NotificationChannelEvent.getGroupUpdated(isNew), 89 channelGroup, uid, pkg, wasBlocked); 90 } 91 92 /** 93 * Log the deletion of a notification channel group. 94 * @param channelGroup The notification channel group. 95 * @param uid UID of app that owns the channel. 96 * @param pkg Package of app that owns the channel. 97 */ logNotificationChannelGroupDeleted(@onNull NotificationChannelGroup channelGroup, int uid, String pkg)98 default void logNotificationChannelGroupDeleted(@NonNull NotificationChannelGroup channelGroup, 99 int uid, String pkg) { 100 logNotificationChannelGroup(NotificationChannelEvent.NOTIFICATION_CHANNEL_GROUP_DELETED, 101 channelGroup, uid, pkg, false); 102 } 103 104 /** 105 * Log blocking or unblocking of the entire app's notifications. 106 * @param uid UID of the app. 107 * @param pkg Package name of the app. 108 * @param enabled If true, notifications are now allowed. 109 */ logAppNotificationsAllowed(int uid, String pkg, boolean enabled)110 default void logAppNotificationsAllowed(int uid, String pkg, boolean enabled) { 111 logAppEvent(NotificationChannelEvent.getBlocked(enabled), uid, pkg); 112 } 113 114 /** 115 * Low-level interface for logging events, to be implemented. 116 * @param event Event to log. 117 * @param channel Notification channel. 118 * @param uid UID of app that owns the channel. 119 * @param pkg Package of app that owns the channel. 120 * @param oldImportance Old importance of the channel, if applicable (0 otherwise). 121 * @param newImportance New importance of the channel, if applicable (0 otherwise). 122 */ logNotificationChannel(@onNull NotificationChannelEvent event, @NonNull NotificationChannel channel, int uid, String pkg, int oldImportance, int newImportance)123 void logNotificationChannel(@NonNull NotificationChannelEvent event, 124 @NonNull NotificationChannel channel, int uid, String pkg, 125 int oldImportance, int newImportance); 126 127 /** 128 * Low-level interface for logging channel group events, to be implemented. 129 * @param event Event to log. 130 * @param channelGroup Notification channel group. 131 * @param uid UID of app that owns the channel. 132 * @param pkg Package of app that owns the channel. 133 * @param wasBlocked True if the channel is being modified and was previously blocked. 134 */ logNotificationChannelGroup(@onNull NotificationChannelEvent event, @NonNull NotificationChannelGroup channelGroup, int uid, String pkg, boolean wasBlocked)135 void logNotificationChannelGroup(@NonNull NotificationChannelEvent event, 136 @NonNull NotificationChannelGroup channelGroup, int uid, String pkg, 137 boolean wasBlocked); 138 139 /** 140 * Low-level interface for logging app-as-a-whole events, to be implemented. 141 * @param uid UID of app. 142 * @param pkg Package of app. 143 */ logAppEvent(@onNull NotificationChannelEvent event, int uid, String pkg)144 void logAppEvent(@NonNull NotificationChannelEvent event, int uid, String pkg); 145 146 /** 147 * The UiEvent enums that this class can log. 148 */ 149 enum NotificationChannelEvent implements UiEventLogger.UiEventEnum { 150 @UiEvent(doc = "App created a new notification channel") 151 NOTIFICATION_CHANNEL_CREATED(219), 152 @UiEvent(doc = "App modified an existing notification channel") 153 NOTIFICATION_CHANNEL_UPDATED(220), 154 @UiEvent(doc = "User modified a new notification channel") 155 NOTIFICATION_CHANNEL_UPDATED_BY_USER(221), 156 @UiEvent(doc = "App deleted an existing notification channel") 157 NOTIFICATION_CHANNEL_DELETED(222), 158 @UiEvent(doc = "App created a new notification channel group") 159 NOTIFICATION_CHANNEL_GROUP_CREATED(223), 160 @UiEvent(doc = "App modified an existing notification channel group") 161 NOTIFICATION_CHANNEL_GROUP_UPDATED(224), 162 @UiEvent(doc = "App deleted an existing notification channel group") 163 NOTIFICATION_CHANNEL_GROUP_DELETED(226), 164 @UiEvent(doc = "System created a new conversation (sub-channel in a notification channel)") 165 NOTIFICATION_CHANNEL_CONVERSATION_CREATED(272), 166 @UiEvent(doc = "System deleted a new conversation (sub-channel in a notification channel)") 167 NOTIFICATION_CHANNEL_CONVERSATION_DELETED(274), 168 @UiEvent(doc = "All notifications for the app were blocked.") 169 APP_NOTIFICATIONS_BLOCKED(557), 170 @UiEvent(doc = "Notifications for the app as a whole were unblocked.") 171 APP_NOTIFICATIONS_UNBLOCKED(558); 172 173 private final int mId; NotificationChannelEvent(int id)174 NotificationChannelEvent(int id) { 175 mId = id; 176 } getId()177 @Override public int getId() { 178 return mId; 179 } 180 getUpdated(boolean byUser)181 public static NotificationChannelEvent getUpdated(boolean byUser) { 182 return byUser 183 ? NotificationChannelEvent.NOTIFICATION_CHANNEL_UPDATED_BY_USER 184 : NotificationChannelEvent.NOTIFICATION_CHANNEL_UPDATED; 185 } 186 getCreated(@onNull NotificationChannel channel)187 public static NotificationChannelEvent getCreated(@NonNull NotificationChannel channel) { 188 return channel.getConversationId() != null 189 ? NotificationChannelEvent.NOTIFICATION_CHANNEL_CONVERSATION_CREATED 190 : NotificationChannelEvent.NOTIFICATION_CHANNEL_CREATED; 191 } 192 getDeleted(@onNull NotificationChannel channel)193 public static NotificationChannelEvent getDeleted(@NonNull NotificationChannel channel) { 194 return channel.getConversationId() != null 195 ? NotificationChannelEvent.NOTIFICATION_CHANNEL_CONVERSATION_DELETED 196 : NotificationChannelEvent.NOTIFICATION_CHANNEL_DELETED; 197 } 198 getGroupUpdated(boolean isNew)199 public static NotificationChannelEvent getGroupUpdated(boolean isNew) { 200 return isNew 201 ? NotificationChannelEvent.NOTIFICATION_CHANNEL_GROUP_CREATED 202 : NotificationChannelEvent.NOTIFICATION_CHANNEL_GROUP_DELETED; 203 } 204 getBlocked(boolean enabled)205 public static NotificationChannelEvent getBlocked(boolean enabled) { 206 return enabled ? APP_NOTIFICATIONS_UNBLOCKED : APP_NOTIFICATIONS_BLOCKED; 207 } 208 } 209 210 /** 211 * @return Small hash of the channel ID, if present, or 0 otherwise. 212 */ getIdHash(@onNull NotificationChannel channel)213 static int getIdHash(@NonNull NotificationChannel channel) { 214 return SmallHash.hash(channel.getId()); 215 } 216 217 /** 218 * @return Small hash of the channel ID, if present, or 0 otherwise. 219 */ getIdHash(@onNull NotificationChannelGroup group)220 static int getIdHash(@NonNull NotificationChannelGroup group) { 221 return SmallHash.hash(group.getId()); 222 } 223 224 /** 225 * @return Logging importance for a channel: the regular importance, or 226 * IMPORTANCE_IMPORTANT_CONVERSATION for a HIGH-importance conversation tagged important. 227 */ getLoggingImportance(@onNull NotificationChannel channel)228 static int getLoggingImportance(@NonNull NotificationChannel channel) { 229 return getLoggingImportance(channel, channel.getImportance()); 230 } 231 232 /** 233 * @return Logging importance for a channel or notification: the regular importance, or 234 * IMPORTANCE_IMPORTANT_CONVERSATION for a HIGH-importance conversation tagged important. 235 */ getLoggingImportance(@onNull NotificationChannel channel, int importance)236 static int getLoggingImportance(@NonNull NotificationChannel channel, int importance) { 237 if (channel.getConversationId() == null || importance < IMPORTANCE_HIGH) { 238 return importance; 239 } 240 return (channel.isImportantConversation()) 241 ? NotificationEnums.IMPORTANCE_IMPORTANT_CONVERSATION 242 : importance; 243 } 244 245 /** 246 * @return "Importance" for a channel group 247 */ getImportance(@onNull NotificationChannelGroup channelGroup)248 static int getImportance(@NonNull NotificationChannelGroup channelGroup) { 249 return getImportance(channelGroup.isBlocked()); 250 } 251 252 /** 253 * @return "Importance" for a channel group, from its blocked status 254 */ getImportance(boolean isBlocked)255 static int getImportance(boolean isBlocked) { 256 return isBlocked 257 ? FrameworkStatsLog.NOTIFICATION_CHANNEL_MODIFIED__IMPORTANCE__IMPORTANCE_NONE 258 : FrameworkStatsLog.NOTIFICATION_CHANNEL_MODIFIED__IMPORTANCE__IMPORTANCE_DEFAULT; 259 } 260 261 } 262