* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Cedar / Account.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Cedar/Account.c b/utvpn/utvpn-unix-v101-7101-public/src/Cedar/Account.c
new file mode 100644 (file)
index 0000000..054df00
--- /dev/null
@@ -0,0 +1,1390 @@
+// SoftEther UT-VPN SourceCode\r
+// \r
+// Copyright (C) 2004-2010 SoftEther Corporation.\r
+// Copyright (C) 2004-2010 University of Tsukuba, Japan.\r
+// Copyright (C) 2003-2010 Daiyuu Nobori.\r
+// All Rights Reserved.\r
+// \r
+// http://utvpn.tsukuba.ac.jp/\r
+// \r
+// This program is free software; you can redistribute it and/or\r
+// modify it under the terms of the GNU General Public License\r
+// version 2 as published by the Free Software Foundation.\r
+// \r
+// This program is distributed in the hope that it will be useful,\r
+// but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
+// GNU General Public License for more details.\r
+// \r
+// You should have received a copy of the GNU General Public License version 2\r
+// along with this program; if not, write to the Free Software\r
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\r
+// \r
+// このファイルは GPL バージョン 2 ライセンスで公開されています。\r
+// 誰でもこのファイルの内容を複製、改変したり、改変したバージョンを再配布\r
+// することができます。ただし、原著作物を改変した場合は、原著作物の著作権表示\r
+// を除去することはできません。改変した著作物を配布する場合は、改変実施者の\r
+// 著作権表示を原著作物の著作権表示に付随して記載するようにしてください。\r
+// \r
+// この SoftEther UT-VPN オープンソース・プロジェクトは、日本国の\r
+// ソフトイーサ株式会社 (SoftEther Corporation, http://www.softether.co.jp/ )\r
+// および筑波大学 (University of Tsukuba, http://www.tsukuba.ac.jp/ ) によって\r
+// ホストされています。\r
+// 本プログラムの配布者は、本プログラムを、業としての利用以外のため、\r
+// および、試験または研究のために利用が行われることを想定して配布\r
+// しています。\r
+// SoftEther UT-VPN プロジェクトの Web サイトは http://utvpn.tsukuba.ac.jp/ に\r
+// あります。\r
+// 本ソフトウェアの不具合の修正、機能改良、セキュリティホールの修復などのコード\r
+// の改変を行った場合で、その成果物を SoftEther UT-VPN プロジェクトに提出して\r
+// いただける場合は、 http://utvpn.tsukuba.ac.jp/ までソースコードを送付して\r
+// ください。SoftEther UT-VPN プロジェクトの本体リリースまたはブランチリリース\r
+// に組み込みさせていただきます。\r
+// \r
+// GPL に基づいて原著作物が提供される本ソフトウェアの改良版を配布、販売する\r
+// 場合は、そのソースコードを GPL に基づいて誰にでも開示する義務が生じます。\r
+// \r
+// 本ソフトウェアに関連する著作権、特許権、商標権はソフトイーサ株式会社\r
+// (SoftEther Corporation) およびその他の著作権保持者が保有しています。\r
+// ソフトイーサ株式会社等はこれらの権利を放棄していません。本ソフトウェアの\r
+// 二次著作物を配布、販売する場合は、これらの権利を侵害しないようにご注意\r
+// ください。\r
+// \r
+// お願い: どのような通信ソフトウェアにも通常は必ず未発見の\r
+// セキュリティホールが潜んでいます。本ソースコードをご覧いただいた結果、\r
+// UT-VPN にセキュリティホールを発見された場合は、当該セキュリティホールの\r
+// 情報を不特定多数に開示される前に、必ず、ソフトイーサ株式会社\r
+// および脆弱性情報の届出を受け付ける公的機関まで通報いただき、\r
+// 公益保護にご協力いただきますようお願い申し上げます。\r
+// \r
+// ソフトイーサ株式会社は、当該セキュリティホールについて迅速に対処を\r
+// 行い、UT-VPN および UT-VPN に関連するソフトウェアのユーザー・顧客\r
+// を保護するための努力を行います。\r
+// \r
+// ソフトイーサへの届出先: http://www.softether.co.jp/jp/contact/\r
+// 日本国内の脆弱性情報届出受付公的機関:\r
+//         独立行政法人 情報処理推進機構\r
+//         http://www.ipa.go.jp/security/vuln/report/\r
+// \r
+// 上記各事項について不明な点は、ソフトイーサ株式会社までご連絡ください。\r
+// 連絡先: http://www.softether.co.jp/jp/contact/\r
+\r
+// -----------------------------------------------\r
+// [ChangeLog]\r
+// 2010.05.20\r
+//  新規リリース by SoftEther\r
+// -----------------------------------------------\r
+\r
+// Account.c\r
+// アカウントマネージャ\r
+\r
+#include "CedarPch.h"\r
+\r
+// ポリシー項目\r
+POLICY_ITEM policy_item[] =\r
+{\r
+//  番号,   数値,   省略可能, 最小, 最大, デフォルト, 単位文字列\r
+// Ver 2.0\r
+       {0,             false,  false,  0,      0,      0,              NULL},                  // Access\r
+       {1,             false,  false,  0,      0,      0,              NULL},                  // DHCPFilter\r
+       {2,             false,  false,  0,      0,      0,              NULL},                  // DHCPNoServer\r
+       {3,             false,  false,  0,      0,      0,              NULL},                  // DHCPForce\r
+       {4,             false,  false,  0,      0,      0,              NULL},                  // NoBridge\r
+       {5,             false,  false,  0,      0,      0,              NULL},                  // NoRouting\r
+       {6,             false,  false,  0,      0,      0,              NULL},                  // CheckMac\r
+       {7,             false,  false,  0,      0,      0,              NULL},                  // CheckIP\r
+       {8,             false,  false,  0,      0,      0,              NULL},                  // ArpDhcpOnly\r
+       {9,             false,  false,  0,      0,      0,              NULL},                  // PrivacyFilter\r
+       {10,    false,  false,  0,      0,      0,              NULL},                  // NoServer\r
+       {11,    false,  false,  0,      0,      0,              NULL},                  // NoBroadcastLimiter\r
+       {12,    false,  false,  0,      0,      0,              NULL},                  // MonitorPort\r
+       {13,    true,   false,  1,      32,     32,             "POL_INT_COUNT"},       // MaxConnection\r
+       {14,    true,   false,  5,      60,     20,             "POL_INT_SEC"}, // TimeOut\r
+       {15,    true,   true,   1,      65535,  0,      "POL_INT_COUNT"},       // MaxMac\r
+       {16,    true,   true,   1,      65535,  0,      "POL_INT_COUNT"},       // MaxIP\r
+       {17,    true,   true,   1,      4294967295UL,   0,      "POL_INT_BPS"}, // MaxUpload\r
+       {18,    true,   true,   1,      4294967295UL,   0,      "POL_INT_BPS"}, // MaxDownload\r
+       {19,    false,  false,  0,      0,      0,              NULL},                  // FixPassword\r
+       {20,    true,   true,   1,      65535,  0,      "POL_INT_COUNT"},       // MultiLogins\r
+       {21,    false,  false,  0,      0,      0,              NULL},                  // NoQoS\r
+// Ver 3.0\r
+       {22,    false,  false,  0,      0,      0,              NULL},                  // RSandRAFilter\r
+       {23,    false,  false,  0,      0,      0,              NULL},                  // RAFilter\r
+       {24,    false,  false,  0,      0,      0,              NULL},                  // DHCPv6Filter\r
+       {25,    false,  false,  0,      0,      0,              NULL},                  // DHCPv6NoServer\r
+       {26,    false,  false,  0,      0,      0,              NULL},                  // NoRoutingV6\r
+       {27,    false,  false,  0,      0,      0,              NULL},                  // CheckIPv6\r
+       {28,    false,  false,  0,      0,      0,              NULL},                  // NoServerV6\r
+       {29,    true,   true,   1,      65535,  0,      "POL_INT_COUNT"},       // MaxIPv6\r
+       {30,    false,  false,  0,      0,      0,              NULL},                  // NoSavePassword\r
+       {31,    true,   true,   1,      4294967295UL,   0,      "POL_INT_SEC"}, // AutoDisconnect\r
+       {32,    false,  false,  0,      0,      0,              NULL},                  // FilterIPv4\r
+       {33,    false,  false,  0,      0,      0,              NULL},                  // FilterIPv6\r
+       {34,    false,  false,  0,      0,      0,              NULL},                  // FilterNonIP\r
+       {35,    false,  false,  0,      0,      0,              NULL},                  // NoIPv6DefaultRouterInRA\r
+       {36,    false,  false,  0,      0,      0,              NULL},                  // NoIPv6DefaultRouterInRAWhenIPv6\r
+       {37,    true,   true,   1,      4095,   0,      "POL_INT_VLAN"},        // VLanId\r
+};\r
+\r
+// ポリシー名を正規化する\r
+char *NormalizePolicyName(char *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       return PolicyIdToStr(PolicyStrToId(name));\r
+}\r
+\r
+// ポリシーの値をフォーマット\r
+void FormatPolicyValue(wchar_t *str, UINT size, UINT id, UINT value)\r
+{\r
+       POLICY_ITEM *p;\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       p = GetPolicyItem(id);\r
+\r
+       if (p->TypeInt == false)\r
+       {\r
+               // bool 型\r
+               if (value == 0)\r
+               {\r
+                       UniStrCpy(str, size, L"No");\r
+               }\r
+               else\r
+               {\r
+                       UniStrCpy(str, size, L"Yes");\r
+               }\r
+       }\r
+       else\r
+       {\r
+               // int 型\r
+               if (value == 0 && p->AllowZero)\r
+               {\r
+                       UniStrCpy(str, size, _UU("CMD_NO_SETTINGS"));\r
+               }\r
+               else\r
+               {\r
+                       UniFormat(str, size, _UU(p->FormatStr), value);\r
+               }\r
+       }\r
+}\r
+\r
+// ポリシーとして設定可能な値の範囲を説明する文字列を取得\r
+void GetPolicyValueRangeStr(wchar_t *str, UINT size, UINT id)\r
+{\r
+       POLICY_ITEM *p;\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       p = GetPolicyItem(id);\r
+\r
+       if (p->TypeInt == false)\r
+       {\r
+               // bool 型\r
+               UniStrCpy(str, size, _UU("CMD_PolicyList_Range_Bool"));\r
+       }\r
+       else\r
+       {\r
+               wchar_t *tag;\r
+               wchar_t tmp1[256], tmp2[256];\r
+\r
+               // int 型\r
+               if (p->AllowZero)\r
+               {\r
+                       tag = _UU("CMD_PolicyList_Range_Int_2");\r
+               }\r
+               else\r
+               {\r
+                       tag = _UU("CMD_PolicyList_Range_Int_1");\r
+               }\r
+\r
+               UniFormat(tmp1, sizeof(tmp1), _UU(p->FormatStr), p->MinValue);\r
+               UniFormat(tmp2, sizeof(tmp2), _UU(p->FormatStr), p->MaxValue);\r
+\r
+               UniFormat(str, size, tag, tmp1, tmp2);\r
+       }\r
+}\r
+\r
+// ポリシーアイテムの取得\r
+POLICY_ITEM *GetPolicyItem(UINT id)\r
+{\r
+       return &policy_item[id];\r
+}\r
+\r
+// 指定されたポリシーがカスケード接続でサポートされているかどうか\r
+bool PolicyIsSupportedForCascade(UINT i)\r
+{\r
+       // このあたりは急いで実装したのでコードがあまり美しくない。\r
+       if (i == 0 || i == 4 || i == 5 || i == 9 || i == 12 || i == 13 ||\r
+               i == 14 || i == 19 || i == 20 || i == 21 || i == 26 || i == 30 || i == 31 || i == 36)\r
+       {\r
+               // これらの項目はカスケード接続でサポートされていない\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// ID をポリシーの名前に変換\r
+char *PolicyIdToStr(UINT i)\r
+{\r
+       // このあたりは急いで実装したのでコードがあまり美しくない。\r
+       switch (i)\r
+       {\r
+       // Ver 2.0\r
+       case 0:         return "Access";\r
+       case 1:         return "DHCPFilter";\r
+       case 2:         return "DHCPNoServer";\r
+       case 3:         return "DHCPForce";\r
+       case 4:         return "NoBridge";\r
+       case 5:         return "NoRouting";\r
+       case 6:         return "CheckMac";\r
+       case 7:         return "CheckIP";\r
+       case 8:         return "ArpDhcpOnly";\r
+       case 9:         return "PrivacyFilter";\r
+       case 10:        return "NoServer";\r
+       case 11:        return "NoBroadcastLimiter";\r
+       case 12:        return "MonitorPort";\r
+       case 13:        return "MaxConnection";\r
+       case 14:        return "TimeOut";\r
+       case 15:        return "MaxMac";\r
+       case 16:        return "MaxIP";\r
+       case 17:        return "MaxUpload";\r
+       case 18:        return "MaxDownload";\r
+       case 19:        return "FixPassword";\r
+       case 20:        return "MultiLogins";\r
+       case 21:        return "NoQoS";\r
+\r
+       // Ver 3.0\r
+       case 22:        return "RSandRAFilter";\r
+       case 23:        return "RAFilter";\r
+       case 24:        return "DHCPv6Filter";\r
+       case 25:        return "DHCPv6NoServer";\r
+       case 26:        return "NoRoutingV6";\r
+       case 27:        return "CheckIPv6";\r
+       case 28:        return "NoServerV6";\r
+       case 29:        return "MaxIPv6";\r
+       case 30:        return "NoSavePassword";\r
+       case 31:        return "AutoDisconnect";\r
+       case 32:        return "FilterIPv4";\r
+       case 33:        return "FilterIPv6";\r
+       case 34:        return "FilterNonIP";\r
+       case 35:        return "NoIPv6DefaultRouterInRA";\r
+       case 36:        return "NoIPv6DefaultRouterInRAWhenIPv6";\r
+       case 37:        return "VLanId";\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// ポリシーの名前を ID に変換\r
+UINT PolicyStrToId(char *name)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return INFINITE;\r
+       }\r
+\r
+       for (i = 0;i < NUM_POLICY_ITEM;i++)\r
+       {\r
+               if (StartWith(PolicyIdToStr(i), name))\r
+               {\r
+                       return i;\r
+               }\r
+       }\r
+\r
+       return INFINITE;\r
+}\r
+\r
+// ポリシーの総数を取得\r
+UINT PolicyNum()\r
+{\r
+       return NUM_POLICY_ITEM;\r
+}\r
+\r
+// 指定した名前をアカウント名として使用できるかどうか確認する\r
+bool IsUserName(char *name)\r
+{\r
+       // このあたりは急いで実装したのでコードがあまり美しくない。\r
+       char tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       StrCpy(tmp, sizeof(tmp), name);\r
+       name = tmp;\r
+\r
+       Trim(name);\r
+\r
+       if (StrLen(name) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, "*") == 0)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (IsSafeStr(name) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, LINK_USER_NAME) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StartWith(name, L3_USERNAME))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, LINK_USER_NAME_PRINT) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, SNAT_USER_NAME) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, SNAT_USER_NAME_PRINT) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, BRIDGE_USER_NAME) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, BRIDGE_USER_NAME_PRINT) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(name, ADMINISTRATOR_USERNAME) == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// ポリシーのタイトルを取得する\r
+wchar_t *GetPolicyTitle(UINT id)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       Format(tmp, sizeof(tmp), "POL_%u", id);\r
+\r
+       return _UU(tmp);\r
+}\r
+\r
+// ポリシーの説明を取得する\r
+wchar_t *GetPolicyDescription(UINT id)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       Format(tmp, sizeof(tmp), "POL_EX_%u", id);\r
+\r
+       return _UU(tmp);\r
+}\r
+\r
+// ポリシーデータのクローン\r
+POLICY *ClonePolicy(POLICY *policy)\r
+{\r
+       POLICY *ret;\r
+       // 引数チェック\r
+       if (policy == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = ZeroMalloc(sizeof(POLICY));\r
+       Copy(ret, policy, sizeof(POLICY));\r
+\r
+       return ret;\r
+}\r
+\r
+// ポリシーを上書きする (古いバージョンを上書きする場合は新しいバージョンのデータは残す)\r
+void OverwritePolicy(POLICY **target, POLICY *p)\r
+{\r
+       // 引数チェック\r
+       if (target == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (p == NULL)\r
+       {\r
+               // ポリシー消去\r
+               if (*target != NULL)\r
+               {\r
+                       Free(*target);\r
+                       *target = NULL;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               if (p->Ver3)\r
+               {\r
+                       // Ver 3\r
+                       if (*target != NULL)\r
+                       {\r
+                               Free(*target);\r
+                               *target = NULL;\r
+                       }\r
+\r
+                       *target = ClonePolicy(p);\r
+               }\r
+               else\r
+               {\r
+                       // Ver 2\r
+                       if (*target == NULL)\r
+                       {\r
+                               *target = ClonePolicy(p);\r
+                       }\r
+                       else\r
+                       {\r
+                               Copy(*target, p, NUM_POLICY_ITEM_FOR_VER2 * sizeof(UINT));\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+// ユーザーポリシーの設定\r
+void SetUserPolicy(USER *u, POLICY *policy)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(u->lock);\r
+       {\r
+               OverwritePolicy(&u->Policy, policy);\r
+       }\r
+       Unlock(u->lock);\r
+}\r
+\r
+// ユーザーポリシーの取得\r
+POLICY *GetUserPolicy(USER *u)\r
+{\r
+       POLICY *ret;\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Lock(u->lock);\r
+       {\r
+               if (u->Policy == NULL)\r
+               {\r
+                       ret = NULL;\r
+               }\r
+               else\r
+               {\r
+                       ret = ClonePolicy(u->Policy);\r
+               }\r
+       }\r
+       Unlock(u->lock);\r
+\r
+       return ret;\r
+}\r
+\r
+// グループポリシーの設定\r
+void SetGroupPolicy(USERGROUP *g, POLICY *policy)\r
+{\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(g->lock);\r
+       {\r
+               OverwritePolicy(&g->Policy, policy);\r
+       }\r
+       Unlock(g->lock);\r
+}\r
+\r
+// グループポリシーの取得\r
+POLICY *GetGroupPolicy(USERGROUP *g)\r
+{\r
+       POLICY *ret;\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Lock(g->lock);\r
+       {\r
+               if (g->Policy == NULL)\r
+               {\r
+                       ret = NULL;\r
+               }\r
+               else\r
+               {\r
+                       ret = ClonePolicy(g->Policy);\r
+               }\r
+       }\r
+       Unlock(g->lock);\r
+\r
+       return ret;\r
+}\r
+\r
+// デフォルトのポリシーを返す\r
+POLICY *GetDefaultPolicy()\r
+{\r
+       // このあたりは急いで実装したのでコードがあまり美しくない。\r
+       static POLICY def_policy =\r
+       {\r
+               true,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               32,\r
+               20,\r
+               0,\r
+               0,\r
+               0,\r
+               0,\r
+               false,\r
+               0,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               0,\r
+               false,\r
+               0,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+               false,\r
+       };\r
+\r
+       return &def_policy;\r
+}\r
+\r
+// NT 認証データの作成\r
+void *NewNTAuthData(wchar_t *username)\r
+{\r
+       AUTHNT *a;\r
+       // 引数チェック\r
+       a = ZeroMallocEx(sizeof(AUTHNT), true);\r
+       a->NtUsername = CopyUniStr(username);\r
+\r
+       return a;\r
+}\r
+\r
+// Radius 認証データの作成\r
+void *NewRadiusAuthData(wchar_t *username)\r
+{\r
+       AUTHRADIUS *a;\r
+       // 引数チェック\r
+       a = ZeroMallocEx(sizeof(AUTHRADIUS), true);\r
+       a->RadiusUsername = CopyUniStr(username);\r
+\r
+       return a;\r
+}\r
+\r
+// ルート証明書による認証データの作成\r
+void *NewRootCertAuthData(X_SERIAL *serial, wchar_t *common_name)\r
+{\r
+       AUTHROOTCERT *a;\r
+\r
+       a = ZeroMallocEx(sizeof(AUTHROOTCERT), true);\r
+       if (common_name != NULL && UniIsEmptyStr(common_name) == false)\r
+       {\r
+               a->CommonName = CopyUniStr(common_name);\r
+       }\r
+       if (serial != NULL && serial->size >= 1)\r
+       {\r
+               a->Serial = CloneXSerial(serial);\r
+       }\r
+\r
+       return a;\r
+}\r
+\r
+// ユーザー証明書認証データの作成\r
+void *NewUserCertAuthData(X *x)\r
+{\r
+       AUTHUSERCERT *a;\r
+\r
+       a = ZeroMalloc(sizeof(AUTHUSERCERT));\r
+       a->UserX = CloneX(x);\r
+\r
+       return a;\r
+}\r
+\r
+// パスワードのハッシュ\r
+void HashPassword(void *dst, char *username, char *password)\r
+{\r
+       BUF *b;\r
+       char *username_upper;\r
+       // 引数チェック\r
+       if (dst == NULL || username == NULL || password == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       b = NewBuf();\r
+       username_upper = CopyStr(username);\r
+       StrUpper(username_upper);\r
+       WriteBuf(b, password, StrLen(password));\r
+       WriteBuf(b, username_upper, StrLen(username_upper));\r
+       Hash(dst, b->Buf, b->Size, true);\r
+\r
+       FreeBuf(b);\r
+       Free(username_upper);\r
+}\r
+\r
+// パスワード認証データの作成\r
+void *NewPasswordAuthData(char *username, char *password)\r
+{\r
+       AUTHPASSWORD *pw;\r
+       // 引数チェック\r
+       if (username == NULL || password == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       pw = ZeroMalloc(sizeof(AUTHPASSWORD));\r
+       HashPassword(pw->HashedKey, username, password);\r
+\r
+       return pw;\r
+}\r
+void *NewPasswordAuthDataRaw(UCHAR *hashed_password)\r
+{\r
+       AUTHPASSWORD *pw;\r
+       // 引数チェック\r
+       if (hashed_password == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       pw = ZeroMalloc(sizeof(AUTHPASSWORD));\r
+       Copy(pw->HashedKey, hashed_password, SHA1_SIZE);\r
+\r
+       return pw;\r
+}\r
+\r
+// ユーザーの認証データのコピー\r
+void *CopyAuthData(void *authdata, UINT authtype)\r
+{\r
+       AUTHPASSWORD *pw = (AUTHPASSWORD *)authdata;\r
+       AUTHUSERCERT *usercert = (AUTHUSERCERT *)authdata;\r
+       AUTHROOTCERT *rootcert = (AUTHROOTCERT *)authdata;\r
+       AUTHRADIUS *radius = (AUTHRADIUS *)authdata;\r
+       AUTHNT *nt = (AUTHNT *)authdata;\r
+       // 引数チェック\r
+       if (authdata == NULL || authtype == AUTHTYPE_ANONYMOUS)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       switch (authtype)\r
+       {\r
+       case AUTHTYPE_PASSWORD:\r
+               {\r
+                       AUTHPASSWORD *ret = ZeroMalloc(sizeof(AUTHPASSWORD));\r
+                       Copy(ret, pw, sizeof(AUTHPASSWORD));\r
+                       return ret;\r
+               }\r
+               break;\r
+\r
+       case AUTHTYPE_USERCERT:\r
+               {\r
+                       AUTHUSERCERT *ret = ZeroMalloc(sizeof(AUTHUSERCERT));\r
+                       ret->UserX = CloneX(usercert->UserX);\r
+                       return ret;\r
+               }\r
+               break;\r
+\r
+       case AUTHTYPE_ROOTCERT:\r
+               {\r
+                       AUTHROOTCERT *ret = ZeroMalloc(sizeof(AUTHROOTCERT));\r
+                       ret->CommonName = CopyUniStr(rootcert->CommonName);\r
+                       ret->Serial = CloneXSerial(rootcert->Serial);\r
+                       return ret;\r
+               }\r
+               break;\r
+\r
+       case AUTHTYPE_RADIUS:\r
+               {\r
+                       AUTHRADIUS *ret = ZeroMalloc(sizeof(AUTHRADIUS));\r
+                       ret->RadiusUsername = UniCopyStr(radius->RadiusUsername);\r
+                       return ret;\r
+               }\r
+               break;\r
+\r
+       case AUTHTYPE_NT:\r
+               {\r
+                       AUTHNT *ret = ZeroMalloc(sizeof(AUTHNT));\r
+                       ret->NtUsername = UniCopyStr(nt->NtUsername);\r
+                       return ret;\r
+               }\r
+               break;\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// ユーザーの認証データのセット\r
+void SetUserAuthData(USER *u, UINT authtype, void *authdata)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (authtype != AUTHTYPE_ANONYMOUS && authdata == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(u->lock);\r
+       {\r
+               if (u->AuthType != AUTHTYPE_ANONYMOUS)\r
+               {\r
+                       // 現在の認証データの解放\r
+                       FreeAuthData(u->AuthType, u->AuthData);\r
+               }\r
+               // 新しい認証データの設定\r
+               u->AuthType = authtype;\r
+               u->AuthData = authdata;\r
+       }\r
+       Unlock(u->lock);\r
+}\r
+\r
+// グループのトラフィックデータを加算\r
+void AddGroupTraffic(USERGROUP *g, TRAFFIC *diff)\r
+{\r
+       // 引数チェック\r
+       if (g == NULL || diff == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(g->lock);\r
+       {\r
+               AddTraffic(g->Traffic, diff);\r
+       }\r
+       Unlock(g->lock);\r
+}\r
+\r
+// ユーザーのトラフィックデータを加算\r
+void AddUserTraffic(USER *u, TRAFFIC *diff)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL || diff == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(u->lock);\r
+       {\r
+               AddTraffic(u->Traffic, diff);\r
+       }\r
+       Unlock(u->lock);\r
+}\r
+\r
+// グループのトラフィック情報をセット\r
+void SetGroupTraffic(USERGROUP *g, TRAFFIC *t)\r
+{\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(g->lock);\r
+       {\r
+               if (t != NULL)\r
+               {\r
+                       Copy(g->Traffic, t, sizeof(TRAFFIC));\r
+               }\r
+               else\r
+               {\r
+                       Zero(g->Traffic, sizeof(TRAFFIC));\r
+               }\r
+       }\r
+       Unlock(g->lock);\r
+}\r
+\r
+// ユーザーのトラフィック情報をセット\r
+void SetUserTraffic(USER *u, TRAFFIC *t)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(u->lock);\r
+       {\r
+               if (t != NULL)\r
+               {\r
+                       Copy(u->Traffic, t, sizeof(TRAFFIC));\r
+               }\r
+               else\r
+               {\r
+                       Zero(u->Traffic, sizeof(TRAFFIC));\r
+               }\r
+       }\r
+       Unlock(u->lock);\r
+}\r
+\r
+// ユーザーをグループに所属させる\r
+void JoinUserToGroup(USER *u, USERGROUP *g)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (g != NULL)\r
+       {\r
+               // 参加\r
+               Lock(u->lock);\r
+               {\r
+                       Lock(g->lock);\r
+                       {\r
+                               if (u->Group != NULL)\r
+                               {\r
+                                       // まずユーザーをグループから外す\r
+                                       ReleaseGroup(u->Group);\r
+                                       u->Group = NULL;\r
+                                       Free(u->GroupName);\r
+                                       u->GroupName = NULL;\r
+                               }\r
+                               // ユーザーをグループに追加する\r
+                               u->GroupName = CopyStr(g->Name);\r
+                               u->Group = g;\r
+                               AddRef(g->ref);\r
+                       }\r
+                       Unlock(g->lock);\r
+               }\r
+               Unlock(u->lock);\r
+       }\r
+       else\r
+       {\r
+               // 脱退\r
+               Lock(u->lock);\r
+               {\r
+                       if (u->Group != NULL)\r
+                       {\r
+                               // ユーザーをグループから外す\r
+                               ReleaseGroup(u->Group);\r
+                               u->Group = NULL;\r
+                               Free(u->GroupName);\r
+                               u->GroupName = NULL;\r
+                       }\r
+               }\r
+               Unlock(u->lock);\r
+       }\r
+}\r
+\r
+// グループ名チェック\r
+bool AcIsGroup(HUB *h, char *name)\r
+{\r
+       USERGROUP *g;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       g = AcGetGroup(h, name);\r
+       if (g == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       ReleaseGroup(g);\r
+\r
+       return true;\r
+}\r
+\r
+// ユーザー名チェック\r
+bool AcIsUser(HUB *h, char *name)\r
+{\r
+       USER *u;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       u = AcGetUser(h, name);\r
+       if (u == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       ReleaseUser(u);\r
+\r
+       return true;\r
+}\r
+\r
+// グループの取得\r
+USERGROUP *AcGetGroup(HUB *h, char *name)\r
+{\r
+       USERGROUP *g, t;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       t.Name = name;\r
+       g = Search(h->HubDb->GroupList, &t);\r
+       if (g == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       AddRef(g->ref);\r
+\r
+       return g;\r
+}\r
+\r
+// ユーザーの取得\r
+USER *AcGetUser(HUB *h, char *name)\r
+{\r
+       USER *u, t;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       t.Name = name;\r
+       u = Search(h->HubDb->UserList, &t);\r
+       if (u == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       AddRef(u->ref);\r
+\r
+       return u;\r
+}\r
+\r
+// ユーザーの削除\r
+bool AcDeleteUser(HUB *h, char *name)\r
+{\r
+       USER *u;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       u = AcGetUser(h, name);\r
+       if (u == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (Delete(h->HubDb->UserList, u))\r
+       {\r
+               ReleaseUser(u);\r
+       }\r
+\r
+       ReleaseUser(u);\r
+\r
+       return true;\r
+}\r
+\r
+// グループの削除\r
+bool AcDeleteGroup(HUB *h, char *name)\r
+{\r
+       USERGROUP *g;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (h == NULL || name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       g = AcGetGroup(h, name);\r
+       if (g == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (Delete(h->HubDb->GroupList, g))\r
+       {\r
+               ReleaseGroup(g);\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(h->HubDb->UserList);i++)\r
+       {\r
+               USER *u = LIST_DATA(h->HubDb->UserList, i);\r
+               Lock(u->lock);\r
+               {\r
+                       if (u->Group == g)\r
+                       {\r
+                               JoinUserToGroup(u, NULL);\r
+                       }\r
+               }\r
+               Unlock(u->lock);\r
+       }\r
+\r
+       ReleaseGroup(g);\r
+\r
+       return true;\r
+}\r
+\r
+// グループの追加\r
+bool AcAddGroup(HUB *h, USERGROUP *g)\r
+{\r
+       // 引数チェック\r
+       if (h == NULL || g == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (LIST_NUM(h->HubDb->GroupList) >= MAX_GROUPS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (AcIsGroup(h, g->Name) != false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Insert(h->HubDb->GroupList, g);\r
+       AddRef(g->ref);\r
+\r
+       return true;\r
+}\r
+\r
+// ユーザーの追加\r
+bool AcAddUser(HUB *h, USER *u)\r
+{\r
+       // 引数チェック\r
+       if (h == NULL || u == NULL || NO_ACCOUNT_DB(h))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (LIST_NUM(h->HubDb->UserList) >= MAX_USERS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (AcIsUser(h, u->Name) != false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Insert(h->HubDb->UserList, u);\r
+       AddRef(u->ref);\r
+\r
+       return true;\r
+}\r
+\r
+// ユーザーの解放\r
+void ReleaseUser(USER *u)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (Release(u->ref) == 0)\r
+       {\r
+               CleanupUser(u);\r
+       }\r
+}\r
+\r
+// ユーザーのクリーンアップ\r
+void CleanupUser(USER *u)\r
+{\r
+       // 引数チェック\r
+       if (u == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       DeleteLock(u->lock);\r
+       Free(u->Name);\r
+       Free(u->RealName);\r
+       Free(u->Note);\r
+       Free(u->GroupName);\r
+       if (u->Group != NULL)\r
+       {\r
+               ReleaseGroup(u->Group);\r
+       }\r
+\r
+       // 認証データの解放\r
+       FreeAuthData(u->AuthType, u->AuthData);\r
+\r
+       if (u->Policy)\r
+       {\r
+               // ポリシー解放\r
+               Free(u->Policy);\r
+       }\r
+\r
+       FreeTraffic(u->Traffic);\r
+\r
+       Free(u);\r
+}\r
+\r
+// 認証データの解放\r
+void FreeAuthData(UINT authtype, void *authdata)\r
+{\r
+       AUTHPASSWORD *pw = (AUTHPASSWORD *)authdata;\r
+       AUTHUSERCERT *uc = (AUTHUSERCERT *)authdata;\r
+       AUTHROOTCERT *rc = (AUTHROOTCERT *)authdata;\r
+       AUTHRADIUS *rd = (AUTHRADIUS *)authdata;\r
+       AUTHNT *nt = (AUTHNT *)authdata;\r
+       // 引数チェック\r
+       if (authtype == AUTHTYPE_ANONYMOUS || authdata == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       switch (authtype)\r
+       {\r
+       case AUTHTYPE_PASSWORD:\r
+               // パスワード認証\r
+               // 何も解放しない\r
+               break;\r
+\r
+       case AUTHTYPE_USERCERT:\r
+               // ユーザー証明書\r
+               FreeX(uc->UserX);\r
+               break;\r
+\r
+       case AUTHTYPE_ROOTCERT:\r
+               // ルート証明書\r
+               if (rc->Serial != NULL)\r
+               {\r
+                       FreeXSerial(rc->Serial);\r
+               }\r
+               if (rc->CommonName != NULL)\r
+               {\r
+                       Free(rc->CommonName);\r
+               }\r
+               break;\r
+\r
+       case AUTHTYPE_RADIUS:\r
+               // Radius 認証\r
+               Free(rd->RadiusUsername);\r
+               break;\r
+\r
+       case AUTHTYPE_NT:\r
+               // Windows NT 認証\r
+               Free(nt->NtUsername);\r
+               break;\r
+       }\r
+\r
+       Free(authdata);\r
+}\r
+\r
+// ユーザーの作成\r
+USER *NewUser(char *name, wchar_t *realname, wchar_t *note, UINT authtype, void *authdata)\r
+{\r
+       USER *u;\r
+       // 引数チェック\r
+       if (name == NULL || realname == NULL || note == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (authtype != AUTHTYPE_ANONYMOUS && authdata == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       u = ZeroMalloc(sizeof(USER));\r
+       u->lock = NewLock();\r
+       u->ref = NewRef();\r
+       u->Name = CopyStr(name);\r
+       u->RealName = CopyUniStr(realname);\r
+       u->Note = CopyUniStr(note);\r
+       u->GroupName = NULL;\r
+       u->Group = NULL;\r
+       u->AuthType = authtype;\r
+       u->AuthData = authdata;\r
+       u->CreatedTime = SystemTime64();\r
+       u->UpdatedTime = SystemTime64();\r
+\r
+       u->Policy = NULL;\r
+       u->Traffic = NewTraffic();\r
+\r
+       return u;\r
+}\r
+\r
+// グループの解放\r
+void ReleaseGroup(USERGROUP *g)\r
+{\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (Release(g->ref) == 0)\r
+       {\r
+               CleanupGroup(g);\r
+       }\r
+}\r
+\r
+// グループのクリーンアップ\r
+void CleanupGroup(USERGROUP *g)\r
+{\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(g->Name);\r
+       Free(g->RealName);\r
+       Free(g->Note);\r
+\r
+       if (g->Policy)\r
+       {\r
+               // ポリシー解放\r
+               Free(g->Policy);\r
+       }\r
+\r
+\r
+       FreeTraffic(g->Traffic);\r
+\r
+       DeleteLock(g->lock);\r
+       Free(g);\r
+}\r
+\r
+// 新しいグループを作成\r
+USERGROUP *NewGroup(char *name, wchar_t *realname, wchar_t *note)\r
+{\r
+       USERGROUP *g;\r
+       // 引数チェック\r
+       if (name == NULL || realname == NULL || note == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       g = ZeroMalloc(sizeof(USERGROUP));\r
+       g->lock = NewLock();\r
+       g->ref = NewRef();\r
+       g->Name = CopyStr(name);\r
+       g->RealName = CopyUniStr(realname);\r
+       g->Note = CopyUniStr(note);\r
+       g->Policy = NULL;\r
+       g->Traffic = NewTraffic();\r
+\r
+       return g;\r
+}\r
+\r
+// HUB のアカウントデータベースのロック\r
+void AcLock(HUB *h)\r
+{\r
+       // 引数チェック\r
+       if (h == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (NO_ACCOUNT_DB(h))\r
+       {\r
+               return;\r
+       }\r
+\r
+       // グループとユーザーをロック\r
+       LockList(h->HubDb->GroupList);\r
+       LockList(h->HubDb->UserList);\r
+}\r
+\r
+// HUB のアカウントデータベースのロック解除\r
+void AcUnlock(HUB *h)\r
+{\r
+       // 引数チェック\r
+       if (h == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (NO_ACCOUNT_DB(h))\r
+       {\r
+               return;\r
+       }\r
+\r
+       // グループとユーザーをロック解除\r
+       UnlockList(h->HubDb->UserList);\r
+       UnlockList(h->HubDb->GroupList);\r
+}\r
+\r
+// グループ名比較関数\r
+int CompareGroupName(void *p1, void *p2)\r
+{\r
+       USERGROUP *g1, *g2;\r
+       // 引数チェック\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       g1 = *(USERGROUP **)p1;\r
+       g2 = *(USERGROUP **)p2;\r
+       if (g1 == NULL || g2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return StrCmpi(g1->Name, g2->Name);\r
+}\r
+\r
+// ユーザー名比較関数\r
+int CompareUserName(void *p1, void *p2)\r
+{\r
+       USER *u1, *u2;\r
+       // 引数チェック\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       u1 = *(USER **)p1;\r
+       u2 = *(USER **)p2;\r
+       if (u1 == NULL || u2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return StrCmpi(u1->Name, u2->Name);\r
+}\r
+\r