Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 1 | /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */ |
| 2 | /* |
| 3 | * Copyright (c) 2013, Regents of the University of California |
| 4 | * Yingdi Yu |
| 5 | * |
| 6 | * BSD license, See the LICENSE file for more information |
| 7 | * |
| 8 | * Author: Yingdi Yu <yingdi@cs.ucla.edu> |
| 9 | */ |
| 10 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 11 | #include "validator-invitation.hpp" |
| 12 | #include "invitation.hpp" |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 13 | |
| 14 | #include "logging.h" |
| 15 | |
Yingdi Yu | eb692ac | 2015-02-10 18:46:18 -0800 | [diff] [blame^] | 16 | namespace chronochat { |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 17 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 18 | using std::vector; |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 19 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 20 | using ndn::CertificateCache; |
| 21 | using ndn::SecRuleRelative; |
| 22 | using ndn::OnDataValidated; |
| 23 | using ndn::OnDataValidationFailed; |
| 24 | using ndn::OnInterestValidated; |
| 25 | using ndn::OnInterestValidationFailed; |
| 26 | using ndn::ValidationRequest; |
| 27 | using ndn::IdentityCertificate; |
Yingdi Yu | 17032f8 | 2014-03-25 15:48:23 -0700 | [diff] [blame] | 28 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 29 | const shared_ptr<CertificateCache> ValidatorInvitation::DefaultCertificateCache = |
| 30 | shared_ptr<CertificateCache>(); |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 31 | |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 32 | ValidatorInvitation::ValidatorInvitation() |
| 33 | : Validator() |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 34 | , m_invitationReplyRule("^([^<CHRONOCHAT-INVITATION>]*)<CHRONOCHAT-INVITATION>", |
| 35 | "^([^<KEY>]*)<KEY>(<>*)[<dsk-.*><ksk-.*>]<ID-CERT>$", |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 36 | "==", "\\1", "\\1\\2", true) |
| 37 | , m_invitationInterestRule("^[^<CHRONOCHAT-INVITATION>]*<CHRONOCHAT-INVITATION><>{6}$") |
| 38 | , m_innerKeyRegex("^([^<KEY>]*)<KEY>(<>*)[<dsk-.*><ksk-.*>]<ID-CERT><>$", "\\1\\2") |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 39 | { |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 40 | } |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 41 | |
| 42 | void |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 43 | ValidatorInvitation::addTrustAnchor(const Name& keyName, const ndn::PublicKey& key) |
| 44 | { |
| 45 | m_trustAnchors[keyName] = key; |
| 46 | } |
| 47 | |
| 48 | void |
| 49 | ValidatorInvitation::removeTrustAnchor(const Name& keyName) |
| 50 | { |
| 51 | m_trustAnchors.erase(keyName); |
| 52 | } |
| 53 | |
| 54 | void |
| 55 | ValidatorInvitation::cleanTrustAnchor() |
| 56 | { |
| 57 | m_trustAnchors.clear(); |
| 58 | } |
| 59 | |
| 60 | void |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 61 | ValidatorInvitation::checkPolicy (const Data& data, |
| 62 | int stepCount, |
| 63 | const OnDataValidated& onValidated, |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 64 | const OnDataValidationFailed& onValidationFailed, |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 65 | vector<shared_ptr<ValidationRequest> >& nextSteps) |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 66 | { |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 67 | const Signature& signature = data.getSignature(); |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 68 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 69 | if (signature.getKeyLocator().getType() != KeyLocator::KeyLocator_Name) |
| 70 | return onValidationFailed(data.shared_from_this(), |
| 71 | "Key Locator is not a name: " + data.getName().toUri()); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 72 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 73 | const Name & keyLocatorName = signature.getKeyLocator().getName(); |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 74 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 75 | if (!m_invitationReplyRule.satisfy(data.getName(), keyLocatorName)) |
| 76 | return onValidationFailed(data.shared_from_this(), |
| 77 | "Does not comply with the invitation rule: " + |
| 78 | data.getName().toUri() + " signed by: " + |
| 79 | keyLocatorName.toUri()); |
| 80 | |
| 81 | Data innerData; |
| 82 | innerData.wireDecode(data.getContent().blockFromValue()); |
| 83 | |
| 84 | return internalCheck(data.wireEncode().value(), |
| 85 | data.wireEncode().value_size() - data.getSignature().getValue().size(), |
| 86 | signature, |
| 87 | keyLocatorName, |
| 88 | innerData, |
| 89 | bind(onValidated, data.shared_from_this()), |
| 90 | bind(onValidationFailed, data.shared_from_this(), _1)); |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 91 | } |
| 92 | |
| 93 | void |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 94 | ValidatorInvitation::checkPolicy (const Interest& interest, |
| 95 | int stepCount, |
| 96 | const OnInterestValidated& onValidated, |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 97 | const OnInterestValidationFailed& onValidationFailed, |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 98 | vector<shared_ptr<ValidationRequest> >& nextSteps) |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 99 | { |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 100 | const Name& interestName = interest.getName(); |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 101 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 102 | if (!m_invitationInterestRule.match(interestName)) |
| 103 | return onValidationFailed(interest.shared_from_this(), |
| 104 | "Invalid interest name: " + interest.getName().toUri()); |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 105 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 106 | Name signedName = interestName.getPrefix(-1); |
Yingdi Yu | 6a61444 | 2014-10-31 17:42:43 -0700 | [diff] [blame] | 107 | ndn::Buffer signedBlob = ndn::Buffer(signedName.wireEncode().value(), |
| 108 | signedName.wireEncode().value_size()); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 109 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 110 | Block signatureBlock = interestName.get(Invitation::SIGNATURE).blockFromValue(); |
| 111 | Block signatureInfo = interestName.get(Invitation::KEY_LOCATOR).blockFromValue(); |
| 112 | Signature signature(signatureInfo, signatureBlock); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 113 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 114 | if (signature.getKeyLocator().getType() != KeyLocator::KeyLocator_Name) |
| 115 | return onValidationFailed(interest.shared_from_this(), |
| 116 | "KeyLocator is not a name: " + interest.getName().toUri()); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 117 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 118 | const Name & keyLocatorName = signature.getKeyLocator().getName(); |
| 119 | |
| 120 | Data innerData; |
| 121 | innerData.wireDecode(interestName.get(Invitation::INVITER_CERT).blockFromValue()); |
| 122 | |
| 123 | return internalCheck(signedBlob.buf(), signedBlob.size(), |
| 124 | signature, |
| 125 | keyLocatorName, |
| 126 | innerData, |
| 127 | bind(onValidated, interest.shared_from_this()), |
| 128 | bind(onValidationFailed, interest.shared_from_this(), _1)); |
Yingdi Yu | fa4ce79 | 2014-02-06 18:09:22 -0800 | [diff] [blame] | 129 | } |
| 130 | |
| 131 | void |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 132 | ValidatorInvitation::internalCheck(const uint8_t* buf, size_t size, |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 133 | const Signature& signature, |
| 134 | const Name& keyLocatorName, |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 135 | const Data& innerData, |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 136 | const OnValidated& onValidated, |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 137 | const OnValidationFailed& onValidationFailed) |
| 138 | { |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 139 | Name signingKeyName = IdentityCertificate::certificateNameToPublicKeyName(keyLocatorName); |
Yingdi Yu | fa0b6a0 | 2014-04-30 14:26:42 -0700 | [diff] [blame] | 140 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 141 | TrustAnchors::const_iterator keyIt = m_trustAnchors.find(signingKeyName); |
| 142 | if (keyIt == m_trustAnchors.end()) |
| 143 | return onValidationFailed("Cannot reach any trust anchor"); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 144 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 145 | if (!Validator::verifySignature(buf, size, signature, keyIt->second)) |
| 146 | return onValidationFailed("Cannot verify outer signature"); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 147 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 148 | // Temporarily disabled, we should get it back when we create a specific key for the chatroom. |
| 149 | // if(!Validator::verifySignature(innerData, m_trustAnchors[signingKeyName])) |
| 150 | // return onValidationFailed("Cannot verify inner signature"); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 151 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 152 | if (!m_innerKeyRegex.match(innerData.getName()) || |
| 153 | m_innerKeyRegex.expand() != signingKeyName.getPrefix(-1)) |
| 154 | return onValidationFailed("Inner certificate does not comply with the rule"); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 155 | |
Yingdi Yu | 0b0a736 | 2014-08-05 16:31:30 -0700 | [diff] [blame] | 156 | return onValidated(); |
Yingdi Yu | 348f5ea | 2014-03-01 14:47:25 -0800 | [diff] [blame] | 157 | } |
| 158 | |
Yingdi Yu | eb692ac | 2015-02-10 18:46:18 -0800 | [diff] [blame^] | 159 | } // namespace chronochat |