• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef CHROME_BROWSER_MAC_KEYSTONE_GLUE_H_
6 #define CHROME_BROWSER_MAC_KEYSTONE_GLUE_H_
7 
8 #include "base/strings/string16.h"
9 
10 #if defined(__OBJC__)
11 
12 #import <Foundation/Foundation.h>
13 
14 #include "base/mac/scoped_authorizationref.h"
15 #import "base/mac/scoped_nsobject.h"
16 
17 // Possible outcomes of various operations.  A version may accompany some of
18 // these, but beware: a version is never required.  For statuses that can be
19 // accompanied by a version, the comment indicates what version is referenced.
20 // A notification posted containing an asynchronous status will always be
21 // followed by a notification with a terminal status.
22 enum AutoupdateStatus {
23   kAutoupdateNone = 0,        // no version (initial state only)
24   kAutoupdateRegistering,     // no version (asynchronous operation in progress)
25   kAutoupdateRegistered,      // no version
26   kAutoupdateChecking,        // no version (asynchronous operation in progress)
27   kAutoupdateCurrent,         // version of the running application
28   kAutoupdateAvailable,       // version of the update that is available
29   kAutoupdateInstalling,      // no version (asynchronous operation in progress)
30   kAutoupdateInstalled,       // version of the update that was installed
31   kAutoupdatePromoting,       // no version (asynchronous operation in progress)
32   kAutoupdatePromoted,        // no version
33   kAutoupdateRegisterFailed,  // no version
34   kAutoupdateCheckFailed,     // no version
35   kAutoupdateInstallFailed,   // no version
36   kAutoupdatePromoteFailed,   // no version
37   kAutoupdateNeedsPromotion,  // no version
38 };
39 
40 // kAutoupdateStatusNotification is the name of the notification posted when
41 // -checkForUpdate and -installUpdate complete.  This notification will be
42 // sent with with its sender object set to the KeystoneGlue instance sending
43 // the notification.  Its userInfo dictionary will contain an AutoupdateStatus
44 // value as an intValue at key kAutoupdateStatusStatus.  If a version is
45 // available (see AutoupdateStatus), it will be present at key
46 // kAutoupdateStatusVersion.
47 extern NSString* const kAutoupdateStatusNotification;
48 extern NSString* const kAutoupdateStatusStatus;
49 extern NSString* const kAutoupdateStatusVersion;
50 
51 namespace {
52 
53 enum BrandFileType {
54   kBrandFileTypeNotDetermined = 0,
55   kBrandFileTypeNone,
56   kBrandFileTypeUser,
57   kBrandFileTypeSystem,
58 };
59 
60 } // namespace
61 
62 // KeystoneGlue is an adapter around the KSRegistration class, allowing it to
63 // be used without linking directly against its containing KeystoneRegistration
64 // framework.  This is used in an environment where most builds (such as
65 // developer builds) don't want or need Keystone support and might not even
66 // have the framework available.  Enabling Keystone support in an application
67 // that uses KeystoneGlue is as simple as dropping
68 // KeystoneRegistration.framework in the application's Frameworks directory
69 // and providing the relevant information in its Info.plist.  KeystoneGlue
70 // requires that the KSUpdateURL key be set in the application's Info.plist,
71 // and that it contain a string identifying the update URL to be used by
72 // Keystone.
73 
74 @class KSRegistration;
75 
76 @interface KeystoneGlue : NSObject {
77  @protected
78 
79   // Data for Keystone registration
80   NSString* productID_;
81   NSString* appPath_;
82   NSString* url_;
83   NSString* version_;
84   NSString* channel_;  // Logically: Dev, Beta, or Stable.
85   BrandFileType brandFileType_;
86 
87   // And the Keystone registration itself, with the active timer
88   KSRegistration* registration_;  // strong
89   NSTimer* timer_;  // strong
90 
91   // The most recent kAutoupdateStatusNotification notification posted.
92   base::scoped_nsobject<NSNotification> recentNotification_;
93 
94   // The authorization object, when it needs to persist because it's being
95   // carried across threads.
96   base::mac::ScopedAuthorizationRef authorization_;
97 
98   // YES if a synchronous promotion operation is in progress (promotion during
99   // installation).
100   BOOL synchronousPromotion_;
101 
102   // YES if an update was ever successfully installed by -installUpdate.
103   BOOL updateSuccessfullyInstalled_;
104 }
105 
106 // Return the default Keystone Glue object.
107 + (id)defaultKeystoneGlue;
108 
109 // Load KeystoneRegistration.framework if present, call into it to register
110 // with Keystone, and set up periodic activity pings.
111 - (void)registerWithKeystone;
112 
113 // -checkForUpdate launches a check for updates, and -installUpdate begins
114 // installing an available update.  For each, status will be communicated via
115 // a kAutoupdateStatusNotification notification, and will also be available
116 // through -recentNotification.
117 - (void)checkForUpdate;
118 - (void)installUpdate;
119 
120 // Accessor for recentNotification_.  Returns an autoreleased NSNotification.
121 - (NSNotification*)recentNotification;
122 
123 // Accessor for the kAutoupdateStatusStatus field of recentNotification_'s
124 // userInfo dictionary.
125 - (AutoupdateStatus)recentStatus;
126 
127 // Returns YES if an asynchronous operation is pending: if an update check or
128 // installation attempt is currently in progress.
129 - (BOOL)asyncOperationPending;
130 
131 // Returns YES if the application is running from a read-only filesystem,
132 // such as a disk image.
133 - (BOOL)isOnReadOnlyFilesystem;
134 
135 // -needsPromotion is YES if the application needs its ticket promoted to
136 // a system ticket.  This will be YES when the application is on a user
137 // ticket and determines that the current user does not have sufficient
138 // permission to perform the update.
139 //
140 // -wantsPromotion is YES if the application wants its ticket promoted to
141 // a system ticket, even if it doesn't need it as determined by
142 // -needsPromotion.  -wantsPromotion will always be YES if -needsPromotion is,
143 // and it will additionally be YES when the application is on a user ticket
144 // and appears to be installed in a system-wide location such as
145 // /Applications.
146 //
147 // Use -needsPromotion to decide whether to show any update UI at all.  If
148 // it's YES, there's no sense in asking the user to "update now" because it
149 // will fail given the rights and permissions involved.  On the other hand,
150 // when -needsPromotion is YES, the application can encourage the user to
151 // promote the ticket so that updates will work properly.
152 //
153 // Use -wantsPromotion to decide whether to allow the user to promote.  The
154 // user shouldn't be nagged about promotion on the basis of -wantsPromotion,
155 // but if it's YES, the user should be allowed to promote the ticket.
156 - (BOOL)needsPromotion;
157 - (BOOL)wantsPromotion;
158 
159 // Promotes the Keystone ticket into the system store.  System Keystone will
160 // be installed if necessary.  If synchronous is NO, the promotion may occur
161 // in the background.  synchronous should be YES for promotion during
162 // installation. The KeystoneGlue object assumes ownership of
163 // authorization_arg.
164 - (void)promoteTicketWithAuthorization:(AuthorizationRef)authorization_arg
165                            synchronous:(BOOL)synchronous;
166 
167 // Requests authorization and calls -promoteTicketWithAuthorization: in
168 // asynchronous mode.
169 - (void)promoteTicket;
170 
171 // Sets a new value for appPath.  Used during installation to point a ticket
172 // at the installed copy.
173 - (void)setAppPath:(NSString*)appPath;
174 
175 @end  // @interface KeystoneGlue
176 
177 @interface KeystoneGlue(ExposedForTesting)
178 
179 // Load any params we need for configuring Keystone.
180 - (void)loadParameters;
181 
182 // Load the Keystone registration object.
183 // Return NO on failure.
184 - (BOOL)loadKeystoneRegistration;
185 
186 - (void)stopTimer;
187 
188 // Called when a checkForUpdate: notification completes.
189 - (void)checkForUpdateComplete:(NSNotification*)notification;
190 
191 // Called when an installUpdate: notification completes.
192 - (void)installUpdateComplete:(NSNotification*)notification;
193 
194 @end  // @interface KeystoneGlue(ExposedForTesting)
195 
196 #endif  // __OBJC__
197 
198 // Functions that may be accessed from non-Objective-C C/C++ code.
199 namespace keystone_glue {
200 
201 // Returns the brand code of the installation. Note that beta, dev, and canary
202 // channels, as well as some stable builds, may have an empty string as a brand
203 // code.
204 std::string BrandCode();
205 
206 // True if Keystone is enabled.
207 bool KeystoneEnabled();
208 
209 // The version of the application currently installed on disk.
210 base::string16 CurrentlyInstalledVersion();
211 
212 }  // namespace keystone_glue
213 
214 #endif  // CHROME_BROWSER_MAC_KEYSTONE_GLUE_H_
215