X509CertificateViewAdapter.java 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * @author masensio
  5. * @author David A. Velasco
  6. * Copyright (C) 2015 ownCloud Inc.
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2,
  10. * as published by the Free Software Foundation.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. *
  20. */
  21. package com.owncloud.android.ui.adapter;
  22. import android.util.Log;
  23. import android.view.View;
  24. import android.widget.TextView;
  25. import com.owncloud.android.R;
  26. import com.owncloud.android.ui.dialog.SslUntrustedCertDialog;
  27. import java.security.MessageDigest;
  28. import java.security.NoSuchAlgorithmException;
  29. import java.security.cert.CertificateEncodingException;
  30. import java.security.cert.X509Certificate;
  31. import java.text.DateFormat;
  32. import java.util.Date;
  33. import java.util.HashMap;
  34. import java.util.Map;
  35. import javax.security.auth.x500.X500Principal;
  36. /**
  37. *
  38. */
  39. public class X509CertificateViewAdapter implements SslUntrustedCertDialog.CertificateViewAdapter {
  40. //private final static String TAG = X509CertificateViewAdapter.class.getSimpleName();
  41. private X509Certificate mCertificate = null;
  42. private static final String TAG = X509CertificateViewAdapter.class.getSimpleName();
  43. public X509CertificateViewAdapter(X509Certificate certificate) {
  44. mCertificate = certificate;
  45. }
  46. @Override
  47. public void updateCertificateView(View dialogView) {
  48. TextView nullCerView = (TextView) dialogView.findViewById(R.id.null_cert);
  49. if (mCertificate != null) {
  50. nullCerView.setVisibility(View.GONE);
  51. showSubject(mCertificate.getSubjectX500Principal(), dialogView);
  52. showIssuer(mCertificate.getIssuerX500Principal(), dialogView);
  53. showValidity(mCertificate.getNotBefore(), mCertificate.getNotAfter(), dialogView);
  54. showSignature(dialogView);
  55. } else {
  56. nullCerView.setVisibility(View.VISIBLE);
  57. }
  58. }
  59. private byte[] getDigest(String algorithm, byte[] message) {
  60. MessageDigest md = null;
  61. try {
  62. md = MessageDigest.getInstance(algorithm);
  63. } catch (NoSuchAlgorithmException e) {
  64. return null;
  65. }
  66. md.reset();
  67. return md.digest(message);
  68. }
  69. private void showSignature(View dialogView) {
  70. byte[] cert = null;
  71. TextView certFingerprintView = ((TextView) dialogView.findViewById(R.id.value_certificate_fingerprint));
  72. TextView algorithmView = ((TextView) dialogView.findViewById(R.id.value_signature_algorithm));
  73. try {
  74. cert = mCertificate.getEncoded();
  75. if (cert == null) {
  76. certFingerprintView.setText(R.string.certificate_load_problem);
  77. algorithmView.setText(R.string.certificate_load_problem);
  78. } else {
  79. certFingerprintView.setText(
  80. getDigestHexBytesWithColonsAndNewLines(dialogView, "SHA-256", cert)
  81. + getDigestHexBytesWithColonsAndNewLines(dialogView, "SHA-1", cert)
  82. + getDigestHexBytesWithColonsAndNewLines(dialogView, "MD5", cert));
  83. algorithmView.setText(mCertificate.getSigAlgName());
  84. }
  85. } catch (CertificateEncodingException e) {
  86. Log.e(TAG, "Problem while trying to decode the certificate.");
  87. }
  88. }
  89. private final String getDigestHexBytesWithColonsAndNewLines(View dialogView, final String digestType, final byte [] cert) {
  90. final byte[] rawDigest;
  91. final String newLine = System.getProperty("line.separator");
  92. rawDigest = getDigest(digestType, cert);
  93. if ( rawDigest == null) {
  94. return digestType + ":" + newLine + dialogView.getContext().getString(R.string.digest_algorithm_not_available) + newLine + newLine;
  95. }
  96. final StringBuilder hex = new StringBuilder(3 * rawDigest.length);
  97. for (final byte b : rawDigest) {
  98. final int hiVal = (b & 0xF0) >> 4;
  99. final int loVal = b & 0x0F;
  100. hex.append((char) ('0' + (hiVal + (hiVal / 10 * 7))));
  101. hex.append((char) ('0' + (loVal + (loVal / 10 * 7))));
  102. hex.append(':');
  103. }
  104. return digestType + ":" + newLine + hex.toString().replaceFirst("\\:$","") + newLine + newLine;
  105. }
  106. private void showValidity(Date notBefore, Date notAfter, View dialogView) {
  107. TextView fromView = ((TextView)dialogView.findViewById(R.id.value_validity_from));
  108. TextView toView = ((TextView)dialogView.findViewById(R.id.value_validity_to));
  109. DateFormat dateFormat = DateFormat.getDateInstance();
  110. fromView.setText(dateFormat.format(notBefore));
  111. toView.setText(dateFormat.format(notAfter));
  112. }
  113. private void showSubject(X500Principal subject, View dialogView) {
  114. Map<String, String> s = parsePrincipal(subject);
  115. TextView cnView = ((TextView)dialogView.findViewById(R.id.value_subject_CN));
  116. TextView oView = ((TextView)dialogView.findViewById(R.id.value_subject_O));
  117. TextView ouView = ((TextView)dialogView.findViewById(R.id.value_subject_OU));
  118. TextView cView = ((TextView)dialogView.findViewById(R.id.value_subject_C));
  119. TextView stView = ((TextView)dialogView.findViewById(R.id.value_subject_ST));
  120. TextView lView = ((TextView)dialogView.findViewById(R.id.value_subject_L));
  121. if (s.get("CN") != null) {
  122. cnView.setText(s.get("CN"));
  123. cnView.setVisibility(View.VISIBLE);
  124. } else {
  125. cnView.setVisibility(View.GONE);
  126. }
  127. if (s.get("O") != null) {
  128. oView.setText(s.get("O"));
  129. oView.setVisibility(View.VISIBLE);
  130. } else {
  131. oView.setVisibility(View.GONE);
  132. }
  133. if (s.get("OU") != null) {
  134. ouView.setText(s.get("OU"));
  135. ouView.setVisibility(View.VISIBLE);
  136. } else {
  137. ouView.setVisibility(View.GONE);
  138. }
  139. if (s.get("C") != null) {
  140. cView.setText(s.get("C"));
  141. cView.setVisibility(View.VISIBLE);
  142. } else {
  143. cView.setVisibility(View.GONE);
  144. }
  145. if (s.get("ST") != null) {
  146. stView.setText(s.get("ST"));
  147. stView.setVisibility(View.VISIBLE);
  148. } else {
  149. stView.setVisibility(View.GONE);
  150. }
  151. if (s.get("L") != null) {
  152. lView.setText(s.get("L"));
  153. lView.setVisibility(View.VISIBLE);
  154. } else {
  155. lView.setVisibility(View.GONE);
  156. }
  157. }
  158. private void showIssuer(X500Principal issuer, View dialogView) {
  159. Map<String, String> s = parsePrincipal(issuer);
  160. TextView cnView = ((TextView)dialogView.findViewById(R.id.value_issuer_CN));
  161. TextView oView = ((TextView)dialogView.findViewById(R.id.value_issuer_O));
  162. TextView ouView = ((TextView)dialogView.findViewById(R.id.value_issuer_OU));
  163. TextView cView = ((TextView)dialogView.findViewById(R.id.value_issuer_C));
  164. TextView stView = ((TextView)dialogView.findViewById(R.id.value_issuer_ST));
  165. TextView lView = ((TextView)dialogView.findViewById(R.id.value_issuer_L));
  166. if (s.get("CN") != null) {
  167. cnView.setText(s.get("CN"));
  168. cnView.setVisibility(View.VISIBLE);
  169. } else {
  170. cnView.setVisibility(View.GONE);
  171. }
  172. if (s.get("O") != null) {
  173. oView.setText(s.get("O"));
  174. oView.setVisibility(View.VISIBLE);
  175. } else {
  176. oView.setVisibility(View.GONE);
  177. }
  178. if (s.get("OU") != null) {
  179. ouView.setText(s.get("OU"));
  180. ouView.setVisibility(View.VISIBLE);
  181. } else {
  182. ouView.setVisibility(View.GONE);
  183. }
  184. if (s.get("C") != null) {
  185. cView.setText(s.get("C"));
  186. cView.setVisibility(View.VISIBLE);
  187. } else {
  188. cView.setVisibility(View.GONE);
  189. }
  190. if (s.get("ST") != null) {
  191. stView.setText(s.get("ST"));
  192. stView.setVisibility(View.VISIBLE);
  193. } else {
  194. stView.setVisibility(View.GONE);
  195. }
  196. if (s.get("L") != null) {
  197. lView.setText(s.get("L"));
  198. lView.setVisibility(View.VISIBLE);
  199. } else {
  200. lView.setVisibility(View.GONE);
  201. }
  202. }
  203. private Map<String, String> parsePrincipal(X500Principal principal) {
  204. Map<String, String> result = new HashMap<String, String>();
  205. String toParse = principal.getName();
  206. String[] pieces = toParse.split(",");
  207. String[] tokens = {"CN", "O", "OU", "C", "ST", "L"};
  208. for (int i=0; i < pieces.length ; i++) {
  209. for (int j=0; j<tokens.length; j++) {
  210. if (pieces[i].startsWith(tokens[j] + "=")) {
  211. result.put(tokens[j], pieces[i].substring(tokens[j].length()+1));
  212. }
  213. }
  214. }
  215. return result;
  216. }
  217. }