Project

General

Profile

Download (26.2 KB) Statistics
| Branch: | Tag: | Revision:

magiccube / src / main / java / org / distorted / network / RubikNetwork.java @ b88cdd91

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2019 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Magic Cube.                                                              //
5
//                                                                                               //
6
// Magic Cube is free software: you can redistribute it and/or modify                            //
7
// it under the terms of the GNU General Public License as published by                          //
8
// the Free Software Foundation, either version 2 of the License, or                             //
9
// (at your option) any later version.                                                           //
10
//                                                                                               //
11
// Magic Cube is distributed in the hope that it will be useful,                                 //
12
// but WITHOUT ANY WARRANTY; without even the implied warranty of                                //
13
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the                                 //
14
// GNU General Public License for more details.                                                  //
15
//                                                                                               //
16
// You should have received a copy of the GNU General Public License                             //
17
// along with Magic Cube.  If not, see <http://www.gnu.org/licenses/>.                           //
18
///////////////////////////////////////////////////////////////////////////////////////////////////
19

    
20
package org.distorted.network;
21

    
22
import java.io.BufferedReader;
23
import java.io.IOException;
24
import java.io.InputStream;
25
import java.io.InputStreamReader;
26
import java.net.HttpURLConnection;
27
import java.net.URL;
28
import java.net.UnknownHostException;
29
import java.security.MessageDigest;
30
import java.security.NoSuchAlgorithmException;
31

    
32
import android.content.pm.PackageInfo;
33
import android.content.pm.PackageManager;
34
import android.graphics.Bitmap;
35
import android.graphics.BitmapFactory;
36

    
37
import androidx.fragment.app.FragmentActivity;
38

    
39
import org.distorted.library.main.DistortedLibrary;
40
import org.distorted.objectlib.json.JsonWriter;
41
import org.distorted.objects.RubikObjectList;
42

    
43
import static org.distorted.objects.RubikObjectList.MAX_LEVEL;
44

    
45
///////////////////////////////////////////////////////////////////////////////////////////////////
46

    
47
public class RubikNetwork
48
  {
49
  public interface Receiver
50
    {
51
    void receive(String[][][] country, String[][][] name, float[][][] time);
52
    void message(String mess);
53
    void error(String error);
54
    }
55

    
56
  public interface Updatee
57
    {
58
    void iconDownloaded(int ordinal, Bitmap bitmap);
59
    void receiveUpdate(RubikUpdates update);
60
    void errorUpdate();
61
    }
62

    
63
  public static final int MAX_PLACES = 10;
64

    
65
  private static final int REND_ADRENO= 0;
66
  private static final int REND_MALI  = 1;
67
  private static final int REND_POWER = 2;
68
  private static final int REND_OTHER = 3;
69

    
70
  private static final int DEBUG_RUNNING = 1;
71
  private static final int DEBUG_SUCCESS = 2;
72
  private static final int DEBUG_FAILURE = 3;
73

    
74
  private final String[] hex = {
75
    "%00", "%01", "%02", "%03", "%04", "%05", "%06", "%07",
76
    "%08", "%09", "%0a", "%0b", "%0c", "%0d", "%0e", "%0f",
77
    "%10", "%11", "%12", "%13", "%14", "%15", "%16", "%17",
78
    "%18", "%19", "%1a", "%1b", "%1c", "%1d", "%1e", "%1f",
79
    "%20", "%21", "%22", "%23", "%24", "%25", "%26", "%27",
80
    "%28", "%29", "%2a", "%2b", "%2c", "%2d", "%2e", "%2f",
81
    "%30", "%31", "%32", "%33", "%34", "%35", "%36", "%37",
82
    "%38", "%39", "%3a", "%3b", "%3c", "%3d", "%3e", "%3f",
83
    "%40", "%41", "%42", "%43", "%44", "%45", "%46", "%47",
84
    "%48", "%49", "%4a", "%4b", "%4c", "%4d", "%4e", "%4f",
85
    "%50", "%51", "%52", "%53", "%54", "%55", "%56", "%57",
86
    "%58", "%59", "%5a", "%5b", "%5c", "%5d", "%5e", "%5f",
87
    "%60", "%61", "%62", "%63", "%64", "%65", "%66", "%67",
88
    "%68", "%69", "%6a", "%6b", "%6c", "%6d", "%6e", "%6f",
89
    "%70", "%71", "%72", "%73", "%74", "%75", "%76", "%77",
90
    "%78", "%79", "%7a", "%7b", "%7c", "%7d", "%7e", "%7f",
91
    "%80", "%81", "%82", "%83", "%84", "%85", "%86", "%87",
92
    "%88", "%89", "%8a", "%8b", "%8c", "%8d", "%8e", "%8f",
93
    "%90", "%91", "%92", "%93", "%94", "%95", "%96", "%97",
94
    "%98", "%99", "%9a", "%9b", "%9c", "%9d", "%9e", "%9f",
95
    "%a0", "%a1", "%a2", "%a3", "%a4", "%a5", "%a6", "%a7",
96
    "%a8", "%a9", "%aa", "%ab", "%ac", "%ad", "%ae", "%af",
97
    "%b0", "%b1", "%b2", "%b3", "%b4", "%b5", "%b6", "%b7",
98
    "%b8", "%b9", "%ba", "%bb", "%bc", "%bd", "%be", "%bf",
99
    "%c0", "%c1", "%c2", "%c3", "%c4", "%c5", "%c6", "%c7",
100
    "%c8", "%c9", "%ca", "%cb", "%cc", "%cd", "%ce", "%cf",
101
    "%d0", "%d1", "%d2", "%d3", "%d4", "%d5", "%d6", "%d7",
102
    "%d8", "%d9", "%da", "%db", "%dc", "%dd", "%de", "%df",
103
    "%e0", "%e1", "%e2", "%e3", "%e4", "%e5", "%e6", "%e7",
104
    "%e8", "%e9", "%ea", "%eb", "%ec", "%ed", "%ee", "%ef",
105
    "%f0", "%f1", "%f2", "%f3", "%f4", "%f5", "%f6", "%f7",
106
    "%f8", "%f9", "%fa", "%fb", "%fc", "%fd", "%fe", "%ff"
107
    };
108

    
109
  private static String[][][] mCountry;
110
  private static String[][][] mName;
111
  private static float[][][] mTime;
112
  private static int[][] mPlaces;
113

    
114
  private static RubikNetwork mThis;
115
  private static String mScores = "";
116
  private static boolean mRunning = false;
117
  private static Receiver mReceiver;
118
  private static Updatee mUpdatee;
119
  private static String mVersion;
120
  private static String mSuspicious;
121
  private static int mNumObjects;
122
  private static RubikUpdates mUpdates;
123
  private static int mDebugState;
124

    
125
///////////////////////////////////////////////////////////////////////////////////////////////////
126

    
127
  private static void initializeStatics()
128
    {
129
    int newNum = RubikObjectList.getNumObjects();
130

    
131
    if( mCountry==null || newNum!=mNumObjects ) mCountry = new String[newNum][MAX_LEVEL][MAX_PLACES];
132
    if( mName==null    || newNum!=mNumObjects ) mName    = new String[newNum][MAX_LEVEL][MAX_PLACES];
133
    if( mTime==null    || newNum!=mNumObjects ) mTime    = new  float[newNum][MAX_LEVEL][MAX_PLACES];
134
    if( mPlaces==null  || newNum!=mNumObjects ) mPlaces  = new    int[newNum][MAX_LEVEL];
135

    
136
    if( mUpdates==null ) mUpdates = new RubikUpdates();
137

    
138
    mNumObjects = newNum;
139
    }
140

    
141
///////////////////////////////////////////////////////////////////////////////////////////////////
142

    
143
  private static String computeHash(String stringToHash, byte[] salt)
144
    {
145
    String generatedPassword;
146

    
147
    try
148
      {
149
      MessageDigest md = MessageDigest.getInstance("MD5");
150
      md.update(salt);
151
      byte[] bytes = md.digest(stringToHash.getBytes());
152
      StringBuilder sb = new StringBuilder();
153

    
154
      for (byte aByte : bytes)
155
        {
156
        sb.append(Integer.toString((aByte & 0xff) + 0x100, 16).substring(1));
157
        }
158

    
159
      generatedPassword = sb.toString();
160
      }
161
    catch (NoSuchAlgorithmException e)
162
      {
163
      return "NoSuchAlgorithm";
164
      }
165

    
166
    return generatedPassword;
167
    }
168

    
169
///////////////////////////////////////////////////////////////////////////////////////////////////
170

    
171
  private boolean fillValuesNormal()
172
    {
173
    int begin=-1 ,end, len = mScores.length();
174
    String row;
175

    
176
    if( len==0 )
177
      {
178
      if( mReceiver!=null ) mReceiver.error("1");
179
      return false;
180
      }
181
    else if( len<=2 )
182
      {
183
      if( mReceiver!=null ) mReceiver.error(mScores);
184
      return false;
185
      }
186

    
187
    for(int i=0; i<mNumObjects; i++)
188
      for(int j=0; j<MAX_LEVEL; j++)
189
        {
190
        mPlaces[i][j] = 0;
191
        }
192

    
193
    while( begin<len )
194
      {
195
      end = mScores.indexOf('\n', begin+1);
196
      if( end<0 ) end = len;
197

    
198
      try
199
        {
200
        row = mScores.substring(begin+1,end);
201
        fillRow(row);
202
        }
203
      catch(Exception ex)
204
        {
205
        // faulty row - ignore
206
        }
207

    
208
      begin = end;
209
      }
210

    
211
    return true;
212
    }
213

    
214
///////////////////////////////////////////////////////////////////////////////////////////////////
215

    
216
  private void fillRow(String row)
217
    {
218
    int s1 = row.indexOf(' ');
219
    int s2 = row.indexOf(' ',s1+1);
220
    int s3 = row.indexOf(' ',s2+1);
221
    int s4 = row.indexOf(' ',s3+1);
222
    int s5 = row.length();
223

    
224
    if( s5>s4 && s4>s3 && s3>s2 && s2>s1 && s1>0 )
225
      {
226
      int object = RubikObjectList.getOrdinal( row.substring(0,s1) );
227

    
228
      if( object>=0 && object<mNumObjects )
229
        {
230
        int level      = Integer.parseInt( row.substring(s1+1,s2) );
231
        String name    = row.substring(s2+1, s3);
232
        int time       = Integer.parseInt( row.substring(s3+1,s4) );
233
        String country = row.substring(s4+1, s5);
234

    
235
        if( country.equals("do") ) country = "dm"; // see RubikScores.setCountry()
236

    
237
        if(level>=0 && level<MAX_LEVEL)
238
          {
239
          int p = mPlaces[object][level];
240
          mPlaces[object][level]++;
241

    
242
          mCountry[object][level][p] = country;
243
          mName   [object][level][p] = name;
244
          mTime   [object][level][p] = ((float)(time/10))/100.0f;
245
          }
246
        }
247
      }
248
    else
249
      {
250
      tryDoCommand(row);
251
      }
252
    }
253

    
254
///////////////////////////////////////////////////////////////////////////////////////////////////
255

    
256
  private void tryDoCommand(String row)
257
    {
258
    if( row.startsWith("comm") )
259
      {
260
      int colon = row.indexOf(':');
261

    
262
      if( colon>0 )
263
        {
264
        String commandNumber = row.substring(4,colon);
265
        int number;
266

    
267
        try
268
          {
269
          number = Integer.parseInt(commandNumber);
270
          }
271
        catch(NumberFormatException ex)
272
          {
273
          number=0;
274
          }
275

    
276
        if(number==1)
277
          {
278
          String country = row.substring(colon+1);
279
          RubikScores scores = RubikScores.getInstance();
280
          scores.setCountry(country);
281
          }
282
        }
283
      }
284
    }
285

    
286
///////////////////////////////////////////////////////////////////////////////////////////////////
287

    
288
  private int getRendererType(String renderer)
289
    {
290
    if( renderer.contains("Adreno")  ) return REND_ADRENO;
291
    if( renderer.contains("Mali")    ) return REND_MALI;
292
    if( renderer.contains("PowerVR") ) return REND_POWER;
293

    
294
    return REND_OTHER;
295
    }
296

    
297
///////////////////////////////////////////////////////////////////////////////////////////////////
298

    
299
  private String parseRenderer(final int type, String renderer)
300
    {
301
    if( type==REND_ADRENO || type==REND_POWER )
302
      {
303
      int lastSpace = renderer.lastIndexOf(' ');
304
      String ret = renderer.substring(lastSpace+1);
305
      return URLencode(ret);
306
      }
307

    
308
    if( type==REND_MALI )
309
      {
310
      int firstHyphen = renderer.indexOf('-');
311
      String ret = renderer.substring(firstHyphen+1);
312
      return URLencode(ret);
313
      }
314

    
315
    return "other";
316
    }
317

    
318
///////////////////////////////////////////////////////////////////////////////////////////////////
319

    
320
  private String parseVersion(final int type, String version)
321
    {
322
    switch(type)
323
      {
324
      case REND_ADRENO: int aMonkey = version.indexOf('@');
325
                        int aDot = version.indexOf('.', aMonkey);
326
                        String ret1 = aDot>=3 ? version.substring(aDot-3,aDot) : "";
327
                        return URLencode(ret1);
328
      case REND_MALI  : int mV1 = version.indexOf("v1");
329
                        int mHyphen = version.indexOf('-', mV1);
330
                        String ret2 = mHyphen>mV1+3 && mV1>=0 ? version.substring(mV1+3,mHyphen) : "";
331
                        return URLencode(ret2);
332
      case REND_POWER : int pMonkey = version.indexOf('@');
333
                        int pSpace  = version.lastIndexOf(' ');
334
                        String ret3 = pSpace>=0 && pMonkey>pSpace+1 ? version.substring(pSpace+1,pMonkey) : "";
335
                        return URLencode(ret3);
336
      default         : return "";
337
      }
338
    }
339

    
340
///////////////////////////////////////////////////////////////////////////////////////////////////
341

    
342
  private String URLencode(String s)
343
    {
344
    StringBuilder sbuf = new StringBuilder();
345
    int len = s.length();
346

    
347
    for (int i = 0; i < len; i++)
348
      {
349
      int ch = s.charAt(i);
350

    
351
           if ('A' <= ch && ch <= 'Z') sbuf.append((char)ch);
352
      else if ('a' <= ch && ch <= 'z') sbuf.append((char)ch);
353
      else if ('0' <= ch && ch <= '9') sbuf.append((char)ch);
354
      else if (ch == ' '             ) sbuf.append('+');
355
      else if (ch == '-' || ch == '_'
356
            || ch == '.' || ch == '!'
357
            || ch == '~' || ch == '*'
358
            || ch == '\'' || ch == '('
359
            || ch == ')'             ) sbuf.append((char)ch);
360
      else if (ch <= 0x007f)           sbuf.append(hex[ch]);
361
      else if (ch <= 0x07FF)
362
        {
363
        sbuf.append(hex[0xc0 | (ch >> 6)]);
364
        sbuf.append(hex[0x80 | (ch & 0x3F)]);
365
        }
366
      else
367
        {
368
        sbuf.append(hex[0xe0 | (ch >> 12)]);
369
        sbuf.append(hex[0x80 | ((ch >> 6) & 0x3F)]);
370
        sbuf.append(hex[0x80 | (ch & 0x3F)]);
371
        }
372
      }
373

    
374
    return sbuf.toString();
375
    }
376

    
377
///////////////////////////////////////////////////////////////////////////////////////////////////
378

    
379
  private boolean network(String url)
380
    {
381
    try
382
      {
383
      java.net.URL connectURL = new URL(url);
384
      HttpURLConnection conn = (HttpURLConnection)connectURL.openConnection();
385

    
386
      conn.setDoInput(true);
387
      conn.setDoOutput(true);
388
      conn.setUseCaches(false);
389
      conn.setRequestMethod("GET");
390
      conn.connect();
391
      conn.getOutputStream().flush();
392

    
393
      InputStream is = conn.getInputStream();
394
      BufferedReader r = new BufferedReader(new InputStreamReader(is));
395
      StringBuilder total = new StringBuilder();
396

    
397
      for (String line; (line = r.readLine()) != null; )
398
        {
399
        total.append(line).append('\n');
400
        }
401

    
402
      mScores = total.toString();
403
      }
404
    catch( final UnknownHostException e )
405
      {
406
      if( mReceiver!=null ) mReceiver.message("No access to Internet");
407
      return false;
408
      }
409
    catch( final SecurityException e )
410
      {
411
      if( mReceiver!=null ) mReceiver.message("Application not authorized to connect to the Internet");
412
      return false;
413
      }
414
    catch( final Exception e )
415
      {
416
      if( mReceiver!=null ) mReceiver.message(e.getMessage());
417
      return false;
418
      }
419

    
420
    if( mScores.length()==0 )
421
      {
422
      if( mReceiver!=null ) mReceiver.message("Failed to download scores");
423
      return false;
424
      }
425

    
426
    return true;
427
    }
428

    
429
///////////////////////////////////////////////////////////////////////////////////////////////////
430

    
431
  private String constructSuspiciousURL()
432
    {
433
    RubikScores scores = RubikScores.getInstance();
434
    int deviceID= scores.getDeviceID();
435
    String suspicious = URLencode(mSuspicious);
436

    
437
    String url="https://distorted.org/magic/cgi-bin/suspicious.cgi";
438
    url += "?i="+deviceID+"&d="+suspicious;
439

    
440
    return url;
441
    }
442

    
443
///////////////////////////////////////////////////////////////////////////////////////////////////
444

    
445
  private String constructDebugURL()
446
    {
447
    RubikScores scores = RubikScores.getInstance();
448
    String name = URLencode(scores.getName());
449
    int numRuns = scores.getNumRuns();
450
    int numPlay = scores.getNumPlays();
451
    String country = scores.getCountry();
452
    String renderer = DistortedLibrary.getDriverRenderer();
453
    String version  = DistortedLibrary.getDriverVersion();
454
    int objectAPI   = JsonWriter.VERSION_OBJECT_MAJOR;
455
    int tutorialAPI = JsonWriter.VERSION_EXTRAS_MAJOR;
456

    
457
    renderer = URLencode(renderer);
458
    version  = URLencode(version);
459

    
460
    String url="https://distorted.org/magic/cgi-bin/debugs-new.cgi";
461
    url += "?n="+name+"&r="+numRuns+"&p="+numPlay+"&c="+country+"&e="+mVersion+"d";
462
    url += "&d="+renderer+"&v="+version+"&a="+objectAPI+"&b="+tutorialAPI;
463

    
464
    return url;
465
    }
466

    
467
///////////////////////////////////////////////////////////////////////////////////////////////////
468

    
469
  private String constructDownloadURL()
470
    {
471
    RubikScores scores = RubikScores.getInstance();
472
    String name = URLencode(scores.getName());
473
    int numRuns = scores.getNumRuns();
474
    int numPlay = scores.getNumPlays();
475
    String country = scores.getCountry();
476

    
477
    String url="https://distorted.org/magic/cgi-bin/download.cgi";
478
    url += "?n="+name+"&r="+numRuns+"&p="+numPlay+"&c="+country+"&e="+mVersion;
479

    
480
    return url;
481
    }
482

    
483
///////////////////////////////////////////////////////////////////////////////////////////////////
484

    
485
  private String constructSubmitURL()
486
    {
487
    RubikScores scores = RubikScores.getInstance();
488
    String name = URLencode(scores.getName());
489
    String veri = scores.isVerified() ? "1" : "";
490
    int numRuns = scores.getNumRuns();
491
    int numPlay = scores.getNumPlays();
492
    int deviceID= scores.getDeviceID();
493
    String reclist = scores.getRecordList("&o=","&l=","&t=");
494
    String country = scores.getCountry();
495
    long epoch = System.currentTimeMillis();
496
    String salt = "cuboid";
497

    
498
    String renderer = DistortedLibrary.getDriverRenderer();
499
    String version  = DistortedLibrary.getDriverVersion();
500

    
501
    int type = getRendererType(renderer);
502
    renderer = parseRenderer(type,renderer);
503
    version  = parseVersion(type,version);
504

    
505
    String url1="https://distorted.org/magic/cgi-bin/submit.cgi";
506
    String url2 = "n="+name+"&v="+veri+"&r="+numRuns+"&p="+numPlay+"&i="+deviceID+"&e="+mVersion;
507
    url2 += "&d="+renderer+"&s="+version+reclist+"&c="+country+"&f="+epoch;
508
    String hash = computeHash( url2, salt.getBytes() );
509

    
510
    return url1 + "?" + url2 + "&h=" + hash;
511
    }
512

    
513
///////////////////////////////////////////////////////////////////////////////////////////////////
514

    
515
  private boolean gottaDownload()
516
    {
517
    return ((mScores.length()==0) && !mRunning);
518
    }
519

    
520
///////////////////////////////////////////////////////////////////////////////////////////////////
521

    
522
  private void figureOutVersion(FragmentActivity act)
523
    {
524
    try
525
      {
526
      PackageInfo pInfo = act.getPackageManager().getPackageInfo( act.getPackageName(), 0);
527
      mVersion = pInfo.versionName;
528
      }
529
    catch (PackageManager.NameNotFoundException e)
530
      {
531
      mVersion = "0.9.2";
532
      }
533
    }
534

    
535
///////////////////////////////////////////////////////////////////////////////////////////////////
536

    
537
  private void downloadThread()
538
    {
539
    try
540
      {
541
      if( gottaDownload() )
542
        {
543
        mRunning = true;
544
        boolean receiveValues = network(constructDownloadURL());
545

    
546
        if( mRunning )
547
          {
548
          receiveValues = fillValuesNormal();
549
          mRunning = false;
550
          }
551

    
552
        if( receiveValues && mReceiver!=null ) mReceiver.receive(mCountry, mName, mTime);
553
        }
554
      }
555
    catch( Exception e )
556
      {
557
      if( mReceiver!=null ) mReceiver.message("Exception downloading records: "+e.getMessage() );
558
      }
559
    }
560

    
561
///////////////////////////////////////////////////////////////////////////////////////////////////
562

    
563
  private void submitThread()
564
    {
565
    try
566
      {
567
      mRunning = true;
568
      RubikScores scores = RubikScores.getInstance();
569

    
570
      if( scores.thereAreUnsubmittedRecords() )
571
        {
572
        boolean receiveValues = network(constructSubmitURL());
573

    
574
        if( mRunning )
575
          {
576
          receiveValues = fillValuesNormal();
577
          mRunning = false;
578
          }
579

    
580
        if( receiveValues )
581
          {
582
          RubikScores.getInstance().successfulSubmit();
583
          if( mReceiver!=null ) mReceiver.receive(mCountry, mName, mTime);
584
          }
585
        }
586
      }
587
    catch( Exception e )
588
      {
589
      if( mReceiver!=null ) mReceiver.message("Exception submitting records: "+e.getMessage() );
590
      }
591
    }
592

    
593
///////////////////////////////////////////////////////////////////////////////////////////////////
594

    
595
  private void debugThread()
596
    {
597
    String url = constructDebugURL();
598
/*
599
    try { Thread.sleep(5000); }
600
    catch( InterruptedException ignored) {}
601
*/
602
    try
603
      {
604
      java.net.URL connectURL = new URL(url);
605
      HttpURLConnection conn = (HttpURLConnection)connectURL.openConnection();
606

    
607
      conn.setDoInput(true);
608
      conn.setDoOutput(true);
609
      conn.setUseCaches(false);
610
      conn.setRequestMethod("GET");
611
      conn.connect();
612
      conn.getOutputStream().flush();
613

    
614
      InputStream is = conn.getInputStream();
615
      BufferedReader r = new BufferedReader(new InputStreamReader(is));
616
      StringBuilder answer = new StringBuilder();
617

    
618
      for (String line; (line = r.readLine()) != null; )
619
        {
620
        answer.append(line).append('\n');
621
        }
622

    
623
      String updates = answer.toString();
624
      mUpdates.parse(updates);
625

    
626
      if( mUpdatee!=null ) mUpdatee.receiveUpdate(mUpdates);
627
      mDebugState = DEBUG_SUCCESS;
628
      }
629
    catch( final Exception e )
630
      {
631
      if( mUpdatee!=null ) mUpdatee.errorUpdate();
632
      mDebugState = DEBUG_FAILURE;
633
      }
634
    }
635

    
636
///////////////////////////////////////////////////////////////////////////////////////////////////
637

    
638
  private void suspiciousThread()
639
    {
640
    String url = constructSuspiciousURL();
641

    
642
    try
643
      {
644
      java.net.URL connectURL = new URL(url);
645
      HttpURLConnection conn = (HttpURLConnection)connectURL.openConnection();
646

    
647
      conn.setDoInput(true);
648
      conn.setDoOutput(true);
649
      conn.setUseCaches(false);
650
      conn.setRequestMethod("GET");
651
      conn.connect();
652
      conn.getOutputStream().flush();
653
      conn.getInputStream();
654
      }
655
    catch( final Exception e )
656
      {
657
      // ignore
658
      }
659
    }
660

    
661
///////////////////////////////////////////////////////////////////////////////////////////////////
662

    
663
  private Bitmap downloadIcon(String url)
664
    {
665
    try
666
      {
667
      java.net.URL connectURL = new URL(url);
668
      HttpURLConnection conn = (HttpURLConnection) connectURL.openConnection();
669
      conn.setDoInput(true);
670
      conn.connect();
671
      InputStream input = conn.getInputStream();
672
      return BitmapFactory.decodeStream(input);
673
      }
674
    catch (IOException e)
675
      {
676
      android.util.Log.e("D", "Failed to download "+url);
677
      android.util.Log.e("D", e.getMessage() );
678
      return null;
679
      }
680
    }
681

    
682
///////////////////////////////////////////////////////////////////////////////////////////////////
683

    
684
  private void iconThread()
685
    {
686
    int numC = mUpdates.getCompletedNumber();
687
    int numS = mUpdates.getStartedNumber();
688

    
689
    for(int c=0; c<numC; c++)
690
      {
691
      Bitmap icon = mUpdates.getCompletedIcon(c);
692

    
693
      if( icon==null )
694
        {
695
        String url = mUpdates.getCompletedURL(c);
696
        icon = downloadIcon(url);
697
        }
698
      if( icon!=null )
699
        {
700
        mUpdates.setCompletedIcon(c,icon);
701
        mUpdatee.iconDownloaded(c,icon);
702
        }
703
      }
704

    
705
    for(int s=0; s<numS; s++)
706
      {
707
      Bitmap icon = mUpdates.getStartedIcon(s);
708

    
709
      if( icon==null )
710
        {
711
        String url = mUpdates.getStartedURL(s);
712
        icon = downloadIcon(url);
713
        }
714
      if( icon!=null )
715
        {
716
        mUpdates.setStartedIcon(s,icon);
717
        mUpdatee.iconDownloaded(numC+s,icon);
718
        }
719
      }
720
    }
721

    
722
///////////////////////////////////////////////////////////////////////////////////////////////////
723

    
724
  private RubikNetwork()
725
    {
726

    
727
    }
728

    
729
///////////////////////////////////////////////////////////////////////////////////////////////////
730
// PUBLIC API
731
///////////////////////////////////////////////////////////////////////////////////////////////////
732

    
733
  public static void onPause()
734
    {
735
    mRunning = false;
736
    }
737

    
738
///////////////////////////////////////////////////////////////////////////////////////////////////
739

    
740
  public static RubikNetwork getInstance()
741
    {
742
    if( mThis==null )
743
      {
744
      mThis = new RubikNetwork();
745
      }
746

    
747
    return mThis;
748
    }
749

    
750
///////////////////////////////////////////////////////////////////////////////////////////////////
751

    
752
  public void download(Receiver receiver, FragmentActivity act)
753
    {
754
    initializeStatics();
755
    mReceiver = receiver;
756
    figureOutVersion(act);
757

    
758
    Thread thread = new Thread()
759
      {
760
      public void run()
761
        {
762
        downloadThread();
763
        }
764
      };
765

    
766
    thread.start();
767
    }
768

    
769
///////////////////////////////////////////////////////////////////////////////////////////////////
770

    
771
  public void submit(Receiver receiver, FragmentActivity act)
772
    {
773
    initializeStatics();
774
    mReceiver = receiver;
775
    figureOutVersion(act);
776

    
777
    Thread thread = new Thread()
778
      {
779
      public void run()
780
        {
781
        submitThread();
782
        }
783
      };
784

    
785
    thread.start();
786
    }
787

    
788
///////////////////////////////////////////////////////////////////////////////////////////////////
789

    
790
  public void debug()
791
    {
792
    initializeStatics();
793
    mDebugState = DEBUG_RUNNING;
794

    
795
    Thread thread = new Thread()
796
      {
797
      public void run()
798
        {
799
        debugThread();
800
        }
801
      };
802

    
803
    thread.start();
804
    }
805

    
806
///////////////////////////////////////////////////////////////////////////////////////////////////
807

    
808
  public void suspicious(String suspicious)
809
    {
810
    initializeStatics();
811
    mSuspicious = suspicious;
812

    
813
    Thread thread = new Thread()
814
      {
815
      public void run()
816
        {
817
        suspiciousThread();
818
        }
819
      };
820

    
821
    thread.start();
822
    }
823

    
824
///////////////////////////////////////////////////////////////////////////////////////////////////
825
// Yes it can happen that the second Updatee registers before we sent an update to the first one
826
// and, as a result, the update never gets sent to the first one. This is not a problem (now, when
827
// there are only two updatees - the RubikStatePlay and the UpdateDialog)
828
//
829
// Yes, there is also a remote possibility that the two threads executing this function and executing
830
// the sendDebug() get swapped exactly in unlucky moment and the update never gets to the updatee.
831
// We don't care about such remote possibility, then the app simply would signal that there are no
832
// updates available.
833

    
834
  public void signUpForUpdates(Updatee updatee)
835
    {
836
         if( mDebugState==DEBUG_SUCCESS ) updatee.receiveUpdate(mUpdates);
837
    else if( mDebugState==DEBUG_FAILURE ) updatee.errorUpdate();
838
    else mUpdatee = updatee;
839
    }
840

    
841
///////////////////////////////////////////////////////////////////////////////////////////////////
842

    
843
  public void downloadIcons(Updatee updatee)
844
    {
845
    initializeStatics();
846
    mUpdatee = updatee;
847

    
848
    Thread thread = new Thread()
849
      {
850
      public void run()
851
        {
852
        iconThread();
853
        }
854
      };
855

    
856
    thread.start();
857
    }
858

    
859
///////////////////////////////////////////////////////////////////////////////////////////////////
860

    
861
  public void updateDone(String shortName)
862
    {
863
    mUpdates.updateDone(shortName);
864
    }
865
}
(1-1/3)