使用SQL Server 2016的Always Encrypt功能防止系統管理員讀取私密性資料
黃家瑞 Jerry Huang
- 精誠資訊/恆逸教育訓練中心-資深講師
- 技術分類:程式設計
在企業中,資訊的讀取是分層級有不同的權限,例如:人事部門主管才能檢視公司員工的人事薪資資料,其他像是產品的成本、客戶資料、廠商資料依據企業屬性不同,常會限制可以讀取或是修改的人員,必須是一定職位的人員才有權限。以往在沒有資訊系統的時代,這些資訊以各種方式進行保護,避免不在該職位的人員窺探這些資訊,然而在導入資訊系統後,這些資料在資料庫管理員(例如:SQL Server 具有sysadmin角色的login user)眼前卻都可以一覽無遺。
資料庫中可以設計密碼機制,讓密碼以密文方式存入資料庫中,使用者驗證時以相同的加密機制,驗證使用者身分的正確性。像是MD5加密的機制便是利用此一理論基礎。但是薪資、員工資料、產品價格、公司資料,這些需要運算或是解密的資訊,如以密文存入資料庫,運算時必須進行解密的運算,除了讓計算效率變得很不理想,也增加使用者的不便性(例如:計算公司薪資時,必須請人事主管輸入加密的密碼,運算時逐筆將薪資的數字解密再做加總)所有資料庫中的彙總函數,合計(SUM)、平均(AVG)、筆數(COUNT)、最大值(MAX)、最小值(MIN)等函數都必須以程式逐筆計算。不僅增加程式設計師的工作負荷,寫出來的系統效能也很差。
SQL Server 2016加入了一項新的功能Always Encrypt,用以解決這個問題:
data:image/s3,"s3://crabby-images/1e067/1e0677bfb673703a2a5301e6343fcb3d12809a06" alt=""
首先準備兩台機器,這裡我以兩部在Hyper-V上的虛擬機器作為範例說明:
- 機器一、DB-Server
- 機器二、AP-Server
- DB-Server
- 防火牆 TCP:1433 Port開啟
- SSCM中TCP/IP通訊協定啟用中
- SSCM SQL Server Browser啟用中
- SQL Server 2016: (為方便講解,使用SQL Server 以及 Windows混合式驗證)
- SSMS 18.0(SSMS 16.5之前的版本僅支援查詢加密的資料,但不支援寫入加密欄位)
- AP-Server
- Visual Studio 2017
步驟一、在DB-Server上建立資料庫
利用以下T-SQL語法建立資料庫db01,新增資料表:
data:image/s3,"s3://crabby-images/c46ec/c46ec94b7b70a5c4992e3d38f37fa2563b95a5cc" alt=""
確認資料可以正常讀取。
步驟二、利用SSMS在使用Always Encrypt
展開Object Explorer => Databases => db01上右鍵 => Encrypt Columns…
data:image/s3,"s3://crabby-images/312fd/312fdef2c87388234e023a0d56e4a3435cf63e5c" alt=""
直接按Next
data:image/s3,"s3://crabby-images/01aa4/01aa45fbf17a884833cf400b6f426ab94ff8532b" alt=""
Encryption Type(加密方式)的設定上,分別在
- employee_name => Randomized
- address => Deterministic
- phone => Randomized
- salary => Deterministic
(Randomized加密方式,資料即使是相同的內容也會產撐出不同的加密文字。Deterministic加密方式,會對相相同料得到相同加密文字。因此,Deterministic的加密欄位還是可以使用DISTINCT、GROUP BY等語法統計有多少不相同資料,Randomized欄位則不支援這些SQL語法。)
data:image/s3,"s3://crabby-images/fc9f2/fc9f2e7db8dd724db7ee83ab67642d3a2e60a6f2" alt=""
在驚嘆號的圖示下可以看一下文字欄位被改為SQL_Latin1_General_CP1_CI_AS定序。
data:image/s3,"s3://crabby-images/78fd9/78fd97b36401a271d807a2139b295b1b5b5cf744" alt=""
SSMS會自動在伺服器上建立憑證,以Master Key Configuration建立成Windows的憑證,接著按一下Next。
data:image/s3,"s3://crabby-images/95fdb/95fdbf13b2dc59c1d9286e67f5b2a1156e4fd465" alt=""
按一下Next執行
data:image/s3,"s3://crabby-images/6f258/6f25839479d4eb473bd261711f1c5093ba402eea" alt=""
SSMS彙總出要執行的動作,按一下Finish開始執行。
data:image/s3,"s3://crabby-images/0a0af/0a0af6f7ecc3c29f57036bc6b15a615f4946046b" alt=""
跑一段時間之後,出現綠色完成的圖示代表完成設定。
data:image/s3,"s3://crabby-images/2504a/2504a198aa34e5132286f3e70ac1c07eb26a1f13" alt=""
步驟三、用SSMS測試一下
在查詢介面中再一次查詢資料。
data:image/s3,"s3://crabby-images/8b627/8b627cf7735ff1d8779b5867ba5445cd7a14baf6" alt=""
此時所看到的文字,就已被加密了。
data:image/s3,"s3://crabby-images/173e6/173e61e07387e202f5147df99834a8d92edf0e6e" alt=""
在SSMS的文字編輯器上按右鍵 => Change Connection => Connection
=> Change Connection
data:image/s3,"s3://crabby-images/4a8bc/4a8bc94ba52bebe9c7ed0ac10d946522e3c5c347" alt=""
切換到Options >>
data:image/s3,"s3://crabby-images/00ece/00ece005fd7e5e297af04b776d35c8ed9160dd12" alt=""
在Additional Connection Parameters的頁籤上加入「Column Encryption Setting = Enabled」,按下 Connect 重新連線。
data:image/s3,"s3://crabby-images/05194/05194c6ab3dbbd705c7be4e043575ed0a89a708a" alt=""
在執行一次查詢的語法
data:image/s3,"s3://crabby-images/e4e99/e4e996aec6b35b2a1f0f55959e20a775fe9bf741" alt=""
這時又可以看到完整的資訊
data:image/s3,"s3://crabby-images/0b941/0b9418745a2c46f070bd058ab190db99156024eb" alt=""
可是如果是新增的語法,以下面語法做測試
data:image/s3,"s3://crabby-images/abac2/abac2a5be4a0ca71b159c4cb260491d58a697227" alt=""
會看到以下的錯誤訊息
data:image/s3,"s3://crabby-images/1c15b/1c15ba902fc0abff752151caf98657745eadaa6d" alt=""
文字編輯器上按右鍵=>Query Options…
data:image/s3,"s3://crabby-images/71feb/71febfc9272a6b61d2769c5b39295de3d98885fb" alt=""
點選 Execution => Advanced 勾選 Enable Parameterization for Always Encrypted
data:image/s3,"s3://crabby-images/1ed98/1ed98e962c437a7bc283d2539ecec3f462f75877" alt=""
使用以下語法執行
data:image/s3,"s3://crabby-images/ede2e/ede2ed4a60fa3144c3686f2ac6356f7a5869b0da" alt=""
此時再一次查詢
data:image/s3,"s3://crabby-images/fd18e/fd18e4981fa6710566c9ae5bd809cd8864407d45" alt=""
資料已經寫入
data:image/s3,"s3://crabby-images/2975b/2975b51aaede8e36e1bc632b1159469369627460" alt=""
步驟四、從DB-Server匯出憑證
在DB-Server虛擬機器上,按下Windows鍵 + Q(快速搜尋),鍵入 cmd 在Command Prompt上按右鍵,執行Run asadministrator。
data:image/s3,"s3://crabby-images/5b06b/5b06b25fd0d7185233f5c06bb4de1558c9bb53fd" alt=""
鍵入 certmgr 按 Enter
data:image/s3,"s3://crabby-images/e26cf/e26cf084758f9d27ac2bbec638c3cf4e07f7278d" alt=""
點選Certificates – Current User => Personal,在 Always Encrypted Auto Certificates上按右鍵=> All Tasks => Export… 匯出憑證
data:image/s3,"s3://crabby-images/23d5d/23d5d1eb5fefb49e35c31ed8700b31cf25ded96c" alt=""
跳出歡迎畫面,直接按 Next 下一步
data:image/s3,"s3://crabby-images/8fc8c/8fc8c57decd4f3d12c61a187b8966ef8729a5c4a" alt=""
選擇 Yes, export the private key 按 Next 跳下一頁
data:image/s3,"s3://crabby-images/b244b/b244ba969166d453aad360830ace009ab37871f4" alt=""
勾選Export all extended properties,按一下 Next
data:image/s3,"s3://crabby-images/1f8e3/1f8e37b66402bb75d11d333f13d7dff8b9c8044b" alt=""
勾選Password,鍵入密碼,按一下 Next
data:image/s3,"s3://crabby-images/e89cf/e89cfbd23b5b94357dea4599233302b00ae369ef" alt=""
選取檔案位置,將憑證匯出到檔案
data:image/s3,"s3://crabby-images/13358/13358b3bcbfa323de76d03f4ad1410068c3fb412" alt=""
按 Finish 完成
data:image/s3,"s3://crabby-images/301d7/301d78f52ee761dc69ad21cba6c3e1ace290620e" alt=""
將憑證從 DB-Server上刪除
data:image/s3,"s3://crabby-images/40060/400604672b11d3406315a772f39eb1b5b60b91f2" alt=""
步驟五、在AP-Server上匯入憑證
在AP-Server上,按下Windows鍵 + Q(快速搜尋),鍵入 cmd 在Command Prompt上按右鍵,執行Run as administrator。
data:image/s3,"s3://crabby-images/33424/3342438c7108c97fa07b2b41a48837804f47ef10" alt=""
鍵入 certmgr 按 Enter
data:image/s3,"s3://crabby-images/6db52/6db5215789e375c834597306a70b002bf3a1f3d7" alt=""
點選Certificates – Current User => Personal 上按右鍵=> All Tasks => Import…
data:image/s3,"s3://crabby-images/bc418/bc418a5217f41551c1dd17636ae20ff3deb0e287" alt=""
直接按 Next
data:image/s3,"s3://crabby-images/3844f/3844f08833b89f5dc61941268771817759753f39" alt=""
匯入由DB-Server憑證匯出的檔案
data:image/s3,"s3://crabby-images/be9cc/be9ccb13a720387fd5784edbdef078882886da5c" alt=""
輸入密碼,按 Next
data:image/s3,"s3://crabby-images/37467/374675a1527a26bbec876d435ba1fce74149504d" alt=""
匯入儲存所在的位置 Personal(預設),按 Next
data:image/s3,"s3://crabby-images/93a6f/93a6ff55fe029215275acdf4a126e46457092b9b" alt=""
按下 Finish 完成匯入
data:image/s3,"s3://crabby-images/a58a2/a58a2761fa89d3bf8f73e83bcc4b35b937946035" alt=""
步驟六、C#程式讀寫資料
撰寫一個簡易的Windows介面程式,畫面設計:
data:image/s3,"s3://crabby-images/4e048/4e048b276df8a61fbc93d7cc86fc684cf9046d6c" alt=""
- Button: btnLoad, btnInsert
- DataGridView: drvData
- TextBox: txtID, txtName, txtAddress, txtPhone, txtSalary
程式碼如下:
data:image/s3,"s3://crabby-images/ee1c5/ee1c5491d8052c262a4bd7d9af4299e054941ad8" alt=""
注意到連線字串
data:image/s3,"s3://crabby-images/8d311/8d3115e105491d31fb8c50d9c6123b8c11419271" alt=""
加入 Column Encryption Setting=Enabled,寫入的SQL語法中
data:image/s3,"s3://crabby-images/3180b/3180bd9e7207881bff6f46b7df43ef8f348d73e5" alt=""
參數的資料型別必須使用和資料表定義中的型別一致。
SQL Server 2016中新增的Always Encrypt讓資料僅能在應用程式端透過權限設計機制讓使用者讀取,避免資料庫管理員可以從資料庫伺服器中直接以T-SQL語法窺探私密性資料。