asp.net - 如何将 Auth0 与现有 ASP.NET Core Identity 数据库结合使用?

标签 asp.net azure authentication asp.net-identity auth0

我正在处理 Auth0应用程序的集成。我有一个现有的应用程序,它使用 ASP.NET Core Identity 和一个 SQL 数据库,当前带有常用表(AspNetUsers、AspNetRoles 等)。

问题是 Auth0 目前具有用于 ASP.NET 成员资格提供程序(MVC3 通用提供程序和 MVC4 简单成员资格)数据库的自定义数据库模板,但没有 ASP.NET Core Identity。由于我不确切知道如何编码与我的 ASP.NET Core Identity 数据库兼容所需的密码哈希,因此这是一个大问题。

有人有用于使用现有 ASP.NET Core Identity 数据库的 Auth0 自定义数据库脚本示例吗?或者,至少是我自己编写的哈希算法?

有关详细信息,ASP.NET Core Identity 数据库有一个 AspNetUsers 表,其中包含此集成感兴趣的以下列:

  • Id(PK,nvarchar,不为空)
  • 电子邮件(nvarchar,null)
  • PasswordHash(nvarchar,null)
  • SecurityStamp(nvarchar,null)
  • 用户名(nvarchar,不为空)

为了清楚起见,我问的是如何在用 JavaScript 编写的 Auth0 配置中设置自定义数据库脚本;不是 ASP.NET Web 应用程序代码。根据文档,网络应用程序的设置非常简单。只是 Auth0 内置自定义数据库模板不包含用于 ASP.NET Core Identity 数据库架构和密码哈希的示例。

任何帮助将不胜感激!

最佳答案

您需要弄清楚所使用的哈希算法并修改模板中的脚本,也使用 Auth0 文档,以使其正确。您可以在 aspnet-identity-pw 中找到该算法的代码。项目。

以下是 Auth0 的示例登录数据库操作脚本,可与存储在 Azure SQL 数据库中的 ASP.NET Core Identity 2.0 数据库配合使用:

function login (username, password, callback) {

  var Connection = require('<a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="285c4d4c41475d5b68190619190618" rel="noreferrer noopener nofollow">[email protected]</a>').Connection;
  var Request = require('<a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="5a2e3f3e33352f291a6b746b6b746a" rel="noreferrer noopener nofollow">[email protected]</a>').Request;
  var TYPES = require('<a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="ef9b8a8b86809a9cafdec1dedec1df" rel="noreferrer noopener nofollow">[email protected]</a>').TYPES;

  var connection = new Connection({
    userName:  configuration.db_username + '@' + configuration.db_server,
    password:  configuration.db_password,
    server:    configuration.db_server, //'dbserver.database.windows.net',
    options: {
      database:  configuration.db_database,
      encrypt: true
    }
  });

  connection.on('debug', function(text) {
    // if you have connection issues, uncomment this to get more detailed info
    //console.log(text);
  }).on('errorMessage', function(text) {
    // this will show any errors when connecting to the SQL database or with the SQL statements
    //console.log(JSON.stringify(text));
  });

  connection.on('connect', function (err) {
    if (err) {
      console.log('error: ' + JSON.stringify(err));
      return callback(err);
    }

    getMembershipUser(username, function(err, user) {
      if (err) {
        return callback(err); // this will return a 500
      }
      if (!user.profile) {
        return callback(); // this will return a 401
      }

      validatePassword(password, user.password.hash, function(err, isValid) {
        if (!isValid) {
          return callback(); // unauthorized
        }

        callback(null, user.profile);
      });

    });
  });


  // Membership Provider implementation used with ASP.NET Core Identity database

  /**
   * getMembershipUser
   *
   * This function gets a username or email and returns a user info, password hashes and salt
   *
   * @usernameOrEamil   {[string]}    the username or email, the method will do a query
   *                                  on both with an OR
   * @callback          {[Function]}  first argument will be the Error if any, and second
   *                                  argument will be a user object
   */
  function getMembershipUser(usernameOrEmail, callback) {
    var user = {};
    var query =
      'SELECT Id, UserName, Email, PasswordHash, SecurityStamp from AspNetUsers ' +
      'WHERE UserName = @UserName';

    var getMembershipQuery = new Request(query);

    getMembershipQuery.addParameter('UserName', TYPES.VarChar, usernameOrEmail);

    getMembershipQuery.on('row', function (fields) {
      user.profile = {};
      user.password = {};
      for(var f in fields) {
        var item = fields[f];
        if (item.metadata.colName === 'Id') {
          user.profile.user_id = item.value;
        } else if (item.metadata.colName === 'UserName') {
          user.profile.nickname = item.value;
        } else if (item.metadata.colName ==='Email') {
          user.profile.email = item.value;
        } else if (item.metadata.colName ==='PasswordHash') {
          user.password.hash = item.value;
        }
      }

      //console.log('User: ' + JSON.stringify(user));
      callback(null, user);
    });

    connection.execSql(getMembershipQuery);
  }

  /**
   * validatePassword
   *
   * This function gets the password entered by the user, and the original password
   * hash and salt from database and performs an HMAC SHA256 hash.
   *
   * @password      {[string]}      the password entered by the user
   * @originalHash  {[string]}      the original password hashed from the database
   *                                (including the salt).
   * @return        {[bool]}        true if password validates
   */
  function validatePassword(password, originalHash, callback) {
    aspnet_identity_pw.validatePassword(password, originalHash, function(result, isValid) {
      console.log('Is Password Valid: ' + isValid);

      callback(null, isValid);
    });
  }

  var aspnet_identity_pw = {
    validatePassword: function(password, hashedPassword, callback) {
      // Original Source:
      //   https://github.com/Syncbak-Git/aspnet-identity-pw/blob/master/lib/aspnet-identity-pw.js
      //   https://www.npmjs.com/package/aspnet-identity-pw
      //   There were some slight modifications to make it run well in Auth0

      var done = false;
      var error = null;
      var result = null;

      if(!hashedPassword) {

          if(callback) {
              callback(null, false);
          }

          return false;
      }

      if(!password) {

          error = new Error("Password is required.");

          if(callback) {
              callback(error);
              return;
          }

          throw error;
      }

      var src = new Buffer(hashedPassword, 'base64');

      if(src.length !== 49 || src[0] !== 0) {
          return false;
      }

      var salt = new Buffer(16);
      src.copy(salt, 0, 1, 17);

      var bytes = new Buffer(32);
      src.copy(bytes, 0, 17, 49);

      var hashed = crypto.pbkdf2Sync(password, salt, 1000, 32, 'sha1');
      result = true;

          for(var i = 0; i < 32; i++) {
              if(bytes[i] !== hashed[i]) {
                  result = false;
                  break;
              }
          }

          done = true;

          if(callback) {
              callback(null, result);
          }

      if(!callback) {
        throw 'callback required!';
      }

      }
  };

  }

这似乎花了很长时间才完全弄清楚。尤其是对密码哈希算法进行编码,直到偶然发现了列出了其代码的 js 项目。

希望这对其他人有帮助!

关于asp.net - 如何将 Auth0 与现有 ASP.NET Core Identity 数据库结合使用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53054123/

相关文章:

Azure DTU 数据库 API

rest - 简单的 REST 身份验证

ruby-on-rails - 如何为 Ruby on Rails 配置 Devise 以将电子邮件和密码存储在用户模型以外的其他位置?

asp.net - 将链接按钮单击从一个用户控件传递到另一个用户控件

c# - ASP.Net DropDownList SelectedIndexChanged 事件触发但不执行任何操作

spring - Azure 服务总线 - Spring Boot 禁用自动启动 (com.microsoft.azure : azure-servicebus-spring-boot-starter)

java - 如何使用 JNDI 在 java 中强制执行 LDAP bindRequest?

asp.net - ETags、IIS7、内核缓存策略(enableKernelCache)

ASP.NET session 意外结束

azure - 如何在本地 U-SQL 数据库中创建 SQL 凭据