| /* |
| * @author: ucla-cs |
| * See COPYING for copyright and distribution information. |
| * This class represents ContentObject Objects |
| */ |
| var ContentObject = function ContentObject(_name,_signedInfo,_content,_signature){ |
| |
| |
| if (typeof _name === 'string'){ |
| this.name = new Name(_name); |
| } |
| else{ |
| //TODO Check the class of _name |
| this.name = _name; |
| } |
| this.signedInfo = _signedInfo; |
| this.content=_content; |
| this.signature = _signature; |
| |
| |
| this.startSIG = null; |
| this.endSIG = null; |
| |
| this.startSignedInfo = null; |
| this.endContent = null; |
| |
| this.rawSignatureData = null; |
| }; |
| |
| ContentObject.prototype.sign = function(){ |
| |
| var n1 = this.encodeObject(this.name); |
| var n2 = this.encodeObject(this.signedInfo); |
| var n3 = this.encodeContent(); |
| |
| var n = n1.concat(n2,n3); |
| |
| if(LOG>2)console.log('Signature Data is (binary) '+n); |
| |
| if(LOG>2)console.log('Signature Data is (RawString)'); |
| |
| if(LOG>2)console.log( DataUtils.toString(n) ); |
| |
| var sig = DataUtils.toString(n); |
| |
| |
| var rsa = new RSAKey(); |
| |
| rsa.readPrivateKeyFromPEMString(globalKeyManager.privateKey); |
| |
| //var hSig = rsa.signString(sig, "sha256"); |
| |
| var hSig = rsa.signByteArrayWithSHA256(n); |
| |
| |
| if(LOG>2)console.log('SIGNATURE SAVED IS'); |
| |
| if(LOG>2)console.log(hSig); |
| |
| if(LOG>2)console.log( DataUtils.toNumbers(hSig.trim())); |
| |
| this.signature.signature = DataUtils.toNumbers(hSig.trim()); |
| |
| |
| }; |
| |
| ContentObject.prototype.encodeObject = function encodeObject(obj){ |
| var enc = new BinaryXMLEncoder(); |
| |
| obj.to_ccnb(enc); |
| |
| var num = enc.getReducedOstream(); |
| |
| return num; |
| |
| |
| }; |
| |
| ContentObject.prototype.encodeContent = function encodeContent(obj){ |
| var enc = new BinaryXMLEncoder(); |
| |
| enc.writeElement(CCNProtocolDTags.Content, this.content); |
| |
| var num = enc.getReducedOstream(); |
| |
| return num; |
| |
| |
| }; |
| |
| ContentObject.prototype.saveRawData = function(bytes){ |
| |
| var sigBits = bytes.slice(this.startSIG, this.endSIG ); |
| |
| this.rawSignatureData = sigBits; |
| }; |
| |
| ContentObject.prototype.from_ccnb = function(/*XMLDecoder*/ decoder) { |
| |
| // TODO VALIDATE THAT ALL FIELDS EXCEPT SIGNATURE ARE PRESENT |
| |
| decoder.readStartElement(this.getElementLabel()); |
| |
| |
| if( decoder.peekStartElement(CCNProtocolDTags.Signature) ){ |
| this.signature = new Signature(); |
| this.signature.from_ccnb(decoder); |
| } |
| |
| //this.endSIG = decoder.offset; |
| |
| this.startSIG = decoder.offset; |
| |
| this.name = new Name(); |
| this.name.from_ccnb(decoder); |
| |
| //this.startSignedInfo = decoder.offset; |
| |
| |
| if( decoder.peekStartElement(CCNProtocolDTags.SignedInfo) ){ |
| this.signedInfo = new SignedInfo(); |
| this.signedInfo.from_ccnb(decoder); |
| } |
| |
| this.content = decoder.readBinaryElement(CCNProtocolDTags.Content); |
| |
| |
| //this.endContent = decoder.offset; |
| this.endSIG = decoder.offset; |
| |
| |
| decoder.readEndElement(); |
| |
| this.saveRawData(decoder.istream); |
| }; |
| |
| ContentObject.prototype.to_ccnb = function(/*XMLEncoder*/ encoder) { |
| |
| //TODO verify name, SignedInfo and Signature is present |
| |
| |
| encoder.writeStartElement(this.getElementLabel()); |
| |
| |
| |
| |
| if(null!=this.signature) this.signature.to_ccnb(encoder); |
| |
| |
| this.startSIG = encoder.offset; |
| |
| |
| if(null!=this.name) this.name.to_ccnb(encoder); |
| |
| //this.endSIG = encoder.offset; |
| //this.startSignedInfo = encoder.offset; |
| |
| |
| if(null!=this.signedInfo) this.signedInfo.to_ccnb(encoder); |
| |
| encoder.writeElement(CCNProtocolDTags.Content, this.content); |
| |
| |
| this.endSIG = encoder.offset; |
| |
| //this.endContent = encoder.offset; |
| |
| |
| encoder.writeEndElement(); |
| |
| this.saveRawData(encoder.ostream); |
| |
| }; |
| |
| ContentObject.prototype.getElementLabel= function(){return CCNProtocolDTags.ContentObject;}; |
| |
| /** |
| * Signature |
| */ |
| var Signature = function Signature(_witness,_signature,_digestAlgorithm) { |
| |
| this.Witness = _witness;//byte [] _witness; |
| this.signature = _signature;//byte [] _signature; |
| this.digestAlgorithm = _digestAlgorithm//String _digestAlgorithm; |
| }; |
| |
| var generateSignature = function(contentName,content,signedinfo){ |
| |
| var enc = new BinaryXMLEncoder(); |
| contentName.to_ccnb(enc); |
| var hex1 = toHex(enc.getReducedOstream()); |
| |
| var enc = new BinaryXMLEncoder(); |
| content.to_ccnb(enc); |
| var hex2 = toHex(enc.getReducedOstream()); |
| |
| var enc = new BinaryXMLEncoder(); |
| signedinfo.to_ccnb(enc); |
| var hex3 = toHex(enc.getReducedOstream()); |
| |
| var hex = hex1+hex2+hex3; |
| |
| //globalKeyManager.sig |
| |
| }; |
| |
| Signature.prototype.from_ccnb =function( decoder) { |
| decoder.readStartElement(this.getElementLabel()); |
| |
| if(LOG>4)console.log('STARTED DECODING SIGNATURE '); |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.DigestAlgorithm)) { |
| |
| if(LOG>4)console.log('DIGIEST ALGORITHM FOUND'); |
| this.digestAlgorithm = decoder.readUTF8Element(CCNProtocolDTags.DigestAlgorithm); |
| } |
| if (decoder.peekStartElement(CCNProtocolDTags.Witness)) { |
| if(LOG>4)console.log('WITNESS FOUND FOUND'); |
| this.Witness = decoder.readBinaryElement(CCNProtocolDTags.Witness); |
| } |
| |
| //FORCE TO READ A SIGNATURE |
| |
| //if(LOG>4)console.log('SIGNATURE FOUND '); |
| this.signature = decoder.readBinaryElement(CCNProtocolDTags.SignatureBits); |
| if(LOG>4)console.log('READ SIGNATURE '); |
| |
| decoder.readEndElement(); |
| |
| }; |
| |
| |
| Signature.prototype.to_ccnb= function( encoder){ |
| |
| if (!this.validate()) { |
| throw new Error("Cannot encode: field values missing."); |
| } |
| |
| encoder.writeStartElement(this.getElementLabel()); |
| |
| if ((null != this.digestAlgorithm) && (!this.digestAlgorithm.equals(CCNDigestHelper.DEFAULT_DIGEST_ALGORITHM))) { |
| encoder.writeElement(CCNProtocolDTags.DigestAlgorithm, OIDLookup.getDigestOID(this.DigestAlgorithm)); |
| } |
| |
| if (null != this.Witness) { |
| // needs to handle null witness |
| encoder.writeElement(CCNProtocolDTags.Witness, this.Witness); |
| } |
| |
| encoder.writeElement(CCNProtocolDTags.SignatureBits, this.signature); |
| |
| encoder.writeEndElement(); |
| }; |
| |
| Signature.prototype.getElementLabel = function() { return CCNProtocolDTags.Signature; }; |
| |
| |
| Signature.prototype.validate = function() { |
| return null != this.signature; |
| }; |
| |
| |
| /** |
| * SignedInfo |
| */ |
| var ContentType = {DATA:0, ENCR:1, GONE:2, KEY:3, LINK:4, NACK:5}; |
| var ContentTypeValue = {0:0x0C04C0, 1:0x10D091,2:0x18E344,3:0x28463F,4:0x2C834A,5:0x34008A}; |
| var ContentTypeValueReverse = {0x0C04C0:0, 0x10D091:1,0x18E344:2,0x28463F:3,0x2C834A:4,0x34008A:5}; |
| |
| var SignedInfo = function SignedInfo(_publisher,_timestamp,_type,_locator,_freshnessSeconds,_finalBlockID){ |
| |
| //TODO, Check types |
| |
| this.publisher = _publisher; //publisherPublicKeyDigest |
| this.timestamp=_timestamp; // CCN Time |
| this.type=_type; // ContentType |
| this.locator =_locator;//KeyLocator |
| this.freshnessSeconds =_freshnessSeconds; // Integer |
| this.finalBlockID=_finalBlockID; //byte array |
| |
| }; |
| |
| SignedInfo.prototype.setFields = function(){ |
| //BASE64 -> RAW STRING |
| |
| //this.locator = new KeyLocator( DataUtils.toNumbersFromString(stringCertificate) ,KeyLocatorType.CERTIFICATE ); |
| |
| var publicKeyHex = globalKeyManager.publicKey; |
| |
| console.log('PUBLIC KEY TO WRITE TO CONTENT OBJECT IS '); |
| console.log(publicKeyHex); |
| |
| var publicKeyBytes = DataUtils.toNumbers(globalKeyManager.publicKey) ; |
| |
| |
| |
| //var stringCertificate = DataUtils.base64toString(globalKeyManager.certificate); |
| |
| //if(LOG>3)console.log('string Certificate is '+stringCertificate); |
| |
| //HEX -> BYTE ARRAY |
| //var publisherkey = DataUtils.toNumbers(hex_sha256(stringCertificate)); |
| |
| //if(LOG>3)console.log('publisher key is '); |
| //if(LOG>3)console.log(publisherkey); |
| |
| var publisherKeyDigest = hex_sha256_from_bytes(publicKeyBytes); |
| |
| this.publisher = new PublisherPublicKeyDigest( DataUtils.toNumbers( publisherKeyDigest ) ); |
| |
| //this.publisher = new PublisherPublicKeyDigest(publisherkey); |
| |
| var d = new Date(); |
| |
| var time = d.getTime(); |
| |
| |
| this.timestamp = new CCNTime( time ); |
| |
| if(LOG>4)console.log('TIME msec is'); |
| |
| if(LOG>4)console.log(this.timestamp.msec); |
| |
| //DATA |
| this.type = 0;//0x0C04C0;//ContentTypeValue[ContentType.DATA]; |
| |
| //if(LOG>4)console.log('toNumbersFromString(stringCertificate) '+DataUtils.toNumbersFromString(stringCertificate)); |
| |
| console.log('PUBLIC KEY TO WRITE TO CONTENT OBJECT IS '); |
| console.log(publicKeyBytes); |
| |
| this.locator = new KeyLocator( publicKeyBytes ,KeyLocatorType.KEY ); |
| |
| //this.locator = new KeyLocator( DataUtils.toNumbersFromString(stringCertificate) ,KeyLocatorType.CERTIFICATE ); |
| |
| }; |
| |
| SignedInfo.prototype.from_ccnb = function( decoder){ |
| |
| decoder.readStartElement( this.getElementLabel() ); |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.PublisherPublicKeyDigest)) { |
| if(LOG>3) console.log('DECODING PUBLISHER KEY'); |
| this.publisher = new PublisherPublicKeyDigest(); |
| this.publisher.from_ccnb(decoder); |
| } |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.Timestamp)) { |
| this.timestamp = decoder.readDateTime(CCNProtocolDTags.Timestamp); |
| if(LOG>4)console.log('TIMESTAMP FOUND IS '+this.timestamp); |
| |
| } |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.Type)) { |
| binType = decoder.readBinaryElement(CCNProtocolDTags.Type);//byte [] |
| |
| |
| //TODO Implement type of Key Reading |
| |
| if(LOG>4)console.log('Binary Type of of Signed Info is '+binType); |
| |
| this.type = binType; |
| |
| |
| //TODO Implement type of Key Reading |
| |
| |
| if (null == this.type) { |
| throw new Error("Cannot parse signedInfo type: bytes."); |
| } |
| |
| } else { |
| this.type = ContentType.DATA; // default |
| } |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.FreshnessSeconds)) { |
| this.freshnessSeconds = decoder.readIntegerElement(CCNProtocolDTags.FreshnessSeconds); |
| if(LOG>4) console.log('FRESHNESS IN SECONDS IS '+ this.freshnessSeconds); |
| } |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.FinalBlockID)) { |
| this.finalBlockID = decoder.readBinaryElement(CCNProtocolDTags.FinalBlockID); |
| } |
| |
| if (decoder.peekStartElement(CCNProtocolDTags.KeyLocator)) { |
| this.locator = new KeyLocator(); |
| this.locator.from_ccnb(decoder); |
| } |
| |
| decoder.readEndElement(); |
| }; |
| |
| SignedInfo.prototype.to_ccnb = function( encoder) { |
| if (!this.validate()) { |
| throw new Error("Cannot encode : field values missing."); |
| } |
| encoder.writeStartElement(this.getElementLabel()); |
| |
| if (null!=this.publisher) { |
| if(LOG>3) console.log('ENCODING PUBLISHER KEY' + this.publisher.publisherPublicKeyDigest); |
| |
| this.publisher.to_ccnb(encoder); |
| } |
| |
| if (null!=this.timestamp) { |
| encoder.writeDateTime(CCNProtocolDTags.Timestamp, this.timestamp ); |
| } |
| |
| if (null!=this.type && this.type !=0) { |
| |
| encoder.writeElement(CCNProtocolDTags.type, this.type); |
| } |
| |
| if (null!=this.freshnessSeconds) { |
| encoder.writeElement(CCNProtocolDTags.FreshnessSeconds, this.freshnessSeconds); |
| } |
| |
| if (null!=this.finalBlockID) { |
| encoder.writeElement(CCNProtocolDTags.FinalBlockID, this.finalBlockID); |
| } |
| |
| if (null!=this.locator) { |
| this.locator.to_ccnb(encoder); |
| } |
| |
| encoder.writeEndElement(); |
| }; |
| |
| SignedInfo.prototype.valueToType = function(){ |
| //for (Entry<byte [], ContentType> entry : ContentValueTypes.entrySet()) { |
| //if (Arrays.equals(value, entry.getKey())) |
| //return entry.getValue(); |
| //} |
| return null; |
| |
| }; |
| |
| SignedInfo.prototype.getElementLabel = function() { |
| return CCNProtocolDTags.SignedInfo; |
| }; |
| |
| SignedInfo.prototype.validate = function() { |
| // We don't do partial matches any more, even though encoder/decoder |
| // is still pretty generous. |
| if (null ==this.publisher || null==this.timestamp ||null== this.locator) |
| return false; |
| return true; |
| }; |