예제 #1
0
    def Start(self):
        """Sign the CSR from the client."""
        with aff4.FACTORY.Create(self.client_id,
                                 aff4_grr.VFSGRRClient,
                                 mode="rw",
                                 token=self.token) as client:

            if self.args.csr.type != rdf_crypto.Certificate.Type.CSR:
                raise IOError("Must be called with CSR")

            csr = rdf_crypto.CertificateSigningRequest(self.args.csr.pem)
            # Verify the CSR. This is not strictly necessary but doesn't harm either.
            try:
                csr.Verify(csr.GetPublicKey())
            except rdf_crypto.VerificationError:
                raise flow.FlowError("CSR for client %s did not verify: %s" %
                                     (self.client_id, csr.AsPEM()))

            # Verify that the CN is of the correct form. The common name should refer
            # to a client URN.
            self.cn = rdf_client.ClientURN.FromPublicKey(csr.GetPublicKey())
            if self.cn != csr.GetCN():
                raise IOError("CSR CN %s does not match public key %s." %
                              (csr.GetCN(), self.cn))

            logging.info("Will sign CSR for: %s", self.cn)

            cert = rdf_crypto.RDFX509Cert.ClientCertFromCSR(csr)

            # This check is important to ensure that the client id reported in the
            # source of the enrollment request is the same as the one in the
            # certificate. We use the ClientURN to ensure this is also of the correct
            # form for a client name.
            if self.cn != self.client_id:
                raise flow.FlowError(
                    "Certificate name %s mismatch for client %s", self.cn,
                    self.client_id)

            # Set and write the certificate to the client record.
            now = rdfvalue.RDFDatetime.Now()
            client.Set(client.Schema.CERT, cert)
            client.Set(client.Schema.FIRST_SEEN, now)
            if data_store.RelationalDBEnabled():
                data_store.REL_DB.WriteClientMetadata(
                    self.client_id.Basename(),
                    certificate=cert,
                    first_seen=now,
                    fleetspeak_enabled=False)

            index = client_index.CreateClientIndex(token=self.token)
            index.AddClient(client)
            if data_store.RelationalDBEnabled:
                index = client_index.ClientIndex()
                index.AddClient(self.client_id.Basename(),
                                data_migration.ConvertVFSGRRClient(client))

        # Publish the client enrollment message.
        self.Publish("ClientEnrollment", self.client_id)

        self.Log("Enrolled %s successfully", self.client_id)
예제 #2
0
    def EnrolFleetspeakClient(self, client_id):
        """Enrols a Fleetspeak-enabled client for use with GRR."""
        client_urn = rdf_client.ClientURN(client_id)

        # If already enrolled, return.
        if data_store.RelationalDBReadEnabled():
            if data_store.REL_DB.ReadClientMetadata(client_id):
                return
        else:
            if aff4.FACTORY.ExistsWithType(client_urn,
                                           aff4_type=aff4_grr.VFSGRRClient,
                                           token=self.token):
                return

        logging.info("Enrolling a new Fleetspeak client: %r", client_id)

        if data_store.RelationalDBWriteEnabled():
            data_store.REL_DB.WriteClientMetadata(client_id,
                                                  fleetspeak_enabled=True)

        # TODO(fleetspeak-team,grr-team): If aff4 isn't reliable enough, we can
        # catch exceptions from it and forward them to Fleetspeak by failing its
        # gRPC call. Fleetspeak will then retry with a random, perhaps healthier,
        # instance of the GRR frontend.
        with aff4.FACTORY.Create(client_urn,
                                 aff4_type=aff4_grr.VFSGRRClient,
                                 mode="rw",
                                 token=self.token) as client:

            client.Set(client.Schema.FLEETSPEAK_ENABLED,
                       rdfvalue.RDFBool(True))

            index = client_index.CreateClientIndex(token=self.token)
            index.AddClient(client)
            if data_store.RelationalDBWriteEnabled():
                index = client_index.ClientIndex()
                index.AddClient(data_migration.ConvertVFSGRRClient(client))

        enrollment_session_id = rdfvalue.SessionID(queue=queues.ENROLLMENT,
                                                   flow_name="Enrol")

        publish_msg = rdf_flows.GrrMessage(
            payload=client_urn,
            session_id=enrollment_session_id,
            # Fleetspeak ensures authentication.
            auth_state=rdf_flows.GrrMessage.AuthorizationState.AUTHENTICATED,
            source=enrollment_session_id,
            priority=rdf_flows.GrrMessage.Priority.MEDIUM_PRIORITY)

        # Publish the client enrollment message.
        events.Events.PublishEvent("ClientEnrollment",
                                   publish_msg,
                                   token=self.token)
예제 #3
0
  def End(self):
    """Finalize client registration."""
    # Update summary and publish to the Discovery queue.
    client = self._OpenClient()
    summary = client.GetSummary()
    self.Publish("Discovery", summary)
    self.SendReply(summary)

    # Update the client index
    client_index.CreateClientIndex(token=self.token).AddClient(client)
    if data_store.RelationalDBEnabled():
      index = client_index.ClientIndex()
      index.AddClient(self.client_id.Basename(),
                      data_migration.ConvertVFSGRRClient(client))