60
60
* The restore token allows the ScreenCast session to be restored
61
61
* with previously granted screen access permissions.
62
62
*/
63
- @ SuppressWarnings ("removal" )
64
63
final class TokenStorage {
65
64
66
65
private TokenStorage () {}
67
66
68
67
private static final String REL_NAME =
68
+ ".java/robot/screencast-tokens.properties" ;
69
+ private static final String REL_NAME_SECONDARY =
69
70
".awt/robot/screencast-tokens.properties" ;
70
71
71
72
private static final Properties PROPS = new Properties ();
72
73
private static final Path PROPS_PATH ;
73
74
private static final Path PROP_FILENAME ;
74
75
76
+ @ SuppressWarnings ("removal" )
75
77
private static void doPrivilegedRunnable (Runnable runnable ) {
76
- AccessController .doPrivileged (new PrivilegedAction <Void >() {
77
- @ Override
78
- public Void run () {
79
- runnable .run ();
80
- return null ;
81
- }
78
+ AccessController .doPrivileged ((PrivilegedAction <Void >) () -> {
79
+ runnable .run ();
80
+ return null ;
82
81
});
83
82
}
84
83
85
84
static {
86
- PROPS_PATH = AccessController .doPrivileged (new PrivilegedAction <Path >() {
87
- @ Override
88
- public Path run () {
89
- return setupPath ();
90
- }
91
- });
85
+ @ SuppressWarnings ("removal" )
86
+ Path propsPath = AccessController
87
+ .doPrivileged ((PrivilegedAction <Path >) () -> setupPath ());
88
+
89
+ PROPS_PATH = propsPath ;
92
90
93
91
if (PROPS_PATH != null ) {
94
92
PROP_FILENAME = PROPS_PATH .getFileName ();
@@ -110,25 +108,32 @@ private static Path setupPath() {
110
108
}
111
109
112
110
Path path = Path .of (userHome , REL_NAME );
111
+ Path secondaryPath = Path .of (userHome , REL_NAME_SECONDARY );
112
+
113
+ boolean copyFromSecondary = !Files .isWritable (path )
114
+ && Files .isWritable (secondaryPath );
115
+
113
116
Path workdir = path .getParent ();
114
117
115
- if (!Files .exists (workdir )) {
116
- try {
117
- Files .createDirectories (workdir );
118
- } catch (Exception e ) {
119
- if (SCREENCAST_DEBUG ) {
120
- System .err .printf ("Token storage: cannot create" +
121
- " directory %s %s\n " , workdir , e );
118
+ if (!Files .isWritable (path )) {
119
+ if (!Files .exists (workdir )) {
120
+ try {
121
+ Files .createDirectories (workdir );
122
+ } catch (Exception e ) {
123
+ if (SCREENCAST_DEBUG ) {
124
+ System .err .printf ("Token storage: cannot create" +
125
+ " directory %s %s\n " , workdir , e );
126
+ }
127
+ return null ;
122
128
}
123
- return null ;
124
129
}
125
- }
126
130
127
- if (!Files .isWritable (workdir )) {
128
- if (SCREENCAST_DEBUG ) {
129
- System .err .printf ("Token storage: %s is not writable\n " , workdir );
131
+ if (!Files .isWritable (workdir )) {
132
+ if (SCREENCAST_DEBUG ) {
133
+ System .err .printf ("Token storage: %s is not writable\n " , workdir );
134
+ }
135
+ return null ;
130
136
}
131
- return null ;
132
137
}
133
138
134
139
try {
@@ -145,7 +150,17 @@ private static Path setupPath() {
145
150
}
146
151
}
147
152
148
- if (Files .exists (path )) {
153
+ if (copyFromSecondary ) {
154
+ if (SCREENCAST_DEBUG ) {
155
+ System .out .println ("Token storage: copying from the secondary location "
156
+ + secondaryPath );
157
+ }
158
+ synchronized (PROPS ) {
159
+ if (readTokens (secondaryPath )) {
160
+ store (path , "copy from the secondary location" );
161
+ }
162
+ }
163
+ } else if (Files .exists (path )) {
149
164
if (!setFilePermission (path )) {
150
165
return null ;
151
166
}
@@ -302,7 +317,7 @@ private static void storeTokenFromNative(String oldToken,
302
317
}
303
318
304
319
if (changed ) {
305
- doPrivilegedRunnable (() -> store ("save tokens" ));
320
+ doPrivilegedRunnable (() -> store (PROPS_PATH , "save tokens" ));
306
321
}
307
322
}
308
323
}
@@ -315,7 +330,7 @@ private static boolean readTokens(Path path) {
315
330
PROPS .clear ();
316
331
PROPS .load (reader );
317
332
}
318
- } catch (IOException e ) {
333
+ } catch (IOException | IllegalArgumentException e ) {
319
334
if (SCREENCAST_DEBUG ) {
320
335
System .err .printf ("""
321
336
Token storage: failed to load property file %s
@@ -410,7 +425,7 @@ static Set<TokenItem> getTokens(List<Rectangle> affectedScreenBounds) {
410
425
}
411
426
412
427
private static void removeMalformedRecords (Set <String > malformedRecords ) {
413
- if (!isWritable ()
428
+ if (!isWritable (PROPS_PATH )
414
429
|| malformedRecords == null
415
430
|| malformedRecords .isEmpty ()) {
416
431
return ;
@@ -424,17 +439,17 @@ private static void removeMalformedRecords(Set<String> malformedRecords) {
424
439
}
425
440
}
426
441
427
- store ("remove malformed records" );
442
+ store (PROPS_PATH , "remove malformed records" );
428
443
}
429
444
}
430
445
431
- private static void store (String failMsg ) {
432
- if (!isWritable ()) {
446
+ private static void store (Path path , String failMsg ) {
447
+ if (!isWritable (path )) {
433
448
return ;
434
449
}
435
450
436
451
synchronized (PROPS ) {
437
- try (BufferedWriter writer = Files .newBufferedWriter (PROPS_PATH )) {
452
+ try (BufferedWriter writer = Files .newBufferedWriter (path )) {
438
453
PROPS .store (writer , null );
439
454
} catch (IOException e ) {
440
455
if (SCREENCAST_DEBUG ) {
@@ -445,13 +460,13 @@ private static void store(String failMsg) {
445
460
}
446
461
}
447
462
448
- private static boolean isWritable () {
449
- if (PROPS_PATH == null
450
- || (Files .exists (PROPS_PATH ) && !Files .isWritable (PROPS_PATH ))) {
463
+ private static boolean isWritable (Path path ) {
464
+ if (path == null
465
+ || (Files .exists (path ) && !Files .isWritable (path ))) {
451
466
452
467
if (SCREENCAST_DEBUG ) {
453
468
System .err .printf (
454
- "Token storage: %s is not writable\n " , PROPS_PATH );
469
+ "Token storage: %s is not writable\n " , path );
455
470
}
456
471
return false ;
457
472
} else {
0 commit comments