WCF安全之ASP.NET兼容模式

本文是利用ASP.NET兼容模式实现WCF安全的一个完整示例,其中用到了ASP.NET的Forms身份验证及Membership,并启用了角色管理。

由于整套安全方案完全利用ASP.NET相关功能实现,而未用到WCF安全策略相关的包括WCF身份验证、WCF授权及WCF传输安全等元素,所以严格的说,这种模式不能算是WCF的安全模式,但该方案确实实现了特定应用场景下的WCF安全。

相比而言,该方案提供的安全程度比WCF的安全策要低一些(例如,未提供全过程的数据传输安全),因此,本方案适应对安全性要求不高的,以IIS为宿主的WCF应用。

本方案中的WCF服务需要以IIS为宿主,可以通过添加“启用Silverlight功能的WCF服务”的方式建立WCF服务。客户端为Silverlight,并在访问WCF服务时使用了Visual Studio 2008自动生成的代理类。

1、建立项目

通过创建“Silverlight应用程序”建立新的项目WcfSecSample,并建立承载该Silverlight的网站WcfSecSample.Web。

2、在Web项目中建立Service目录,并在该目录下添加WCF服务WeatherService,服务类的完整代码如下:

using System;
using System.Linq;
using System.Runtime.Serialization;
using System.Security.Principal;
using System.ServiceModel;
using System.ServiceModel.Activation;
using System.Collections.Generic;
using System.Text;
using System.Web;
using System.Web.Security;namespace WcfSecSample.Web
{[ServiceContract(Namespace = "")][AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]public class WeatherService{private static string s_weather = "Sunny";[OperationContract]public void SetWeather(string weather){if (!HttpContext.Current.User.IsInRole("Admin")) throw new ApplicationException("无权限。");s_weather = weather;}[OperationContract]public string GetWeather(){if (!HttpContext.Current.User.IsInRole("Guest")) throw new ApplicationException("无权限。");return s_weather;}}
}

3、在Web项目中添加用于登录的WCF服务LoginService,该服务类的完整代码如下:

using System;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.ServiceModel.Activation;
using System.Collections.Generic;
using System.Text;
using System.Web.Security;namespace WcfSecSample.Web
{[ServiceContract(Namespace = "")][AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]public class LoginService{[OperationContract]public bool Login(string userName, string password){bool isValid = Membership.ValidateUser(userName, password);if (isValid) FormsAuthentication.SetAuthCookie(userName, false);return isValid;}[OperationContract]public void SignOut(){FormsAuthentication.SignOut();}}
}

4、在Web项目中添加继承自MembershipProvider类的CustomerMembershipProvider类,暂时只实现了本方案所需要的ValidateUser方法。ValidateUser方法的代码如下:

        public override bool ValidateUser(string username, string password){return username == "admin" && password == "123456" || username == "guest";}

5、在Web项目中添加继承自RoleProvider类的CustomRoleProvider类,暂时只实现了本方案所需要的GetRolesForUser方法。GetRolesForUser方法的代码如下:

        public override string[] GetRolesForUser(string username){if (username == "admin") return new []{"Admin", "Guest"};return new[] { "Guest" };}

需要注意的是,虽然凭感觉HttpContext.Current.User.IsInRole方法应该最终调用RoleProvider类的IsUserInRole方法,但事实却是最终调用了RoleProvider类的GetRolesForUser方法完成的。if (!HttpContext.Current.User.IsInRole("Admin"))还可以换成if (Roles.IsUserInRole(HttpContext.Current.User.Identity.Name, "Admin")),同样是最终调用了RoleProvider类的GetRolesForUser方法。

6、配置Web.config文件

在system.web节内添加如下内容:

    <authentication mode="Forms" ><forms name=".sec" loginUrl="LoginService.svc"></forms></authentication><membership defaultProvider="default"><providers><add name="default" type="WcfSecSample.Web.CustomerMembershipProvider, WcfSecSample.Web"/></providers></membership><roleManager defaultProvider="default" enabled="true"><providers><add name="default" type="WcfSecSample.Web.CustomRoleProvider, WcfSecSample.Web"/></providers></roleManager>

以上三节内容分别配置了身份验证模式、MembershipProvider及RoleProvider。

然后在示例服务所在的Service目录下添加Web.config文件,禁止对Service目录的匿名访问。该文件的内容如下:

<?xml version="1.0" encoding="utf-8"?>
<configuration><system.web><authorization><deny users="?"/></authorization></system.web>
</configuration>

至此,服务端的工作就完成了,接下来建立客户端测试示例。

7、添加服务引用。

在Silverlight项目中添加对WeatherService的服务引用WeatherServiceRef。需注意的是进行该操作时需要暂时允许对Service目录的匿名访问。

8、在MainPage中添加测试代码。完成之后的代码如下:

MainPage.xaml文件:

<UserControl x:Class="WcfSecSample.MainPage"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" d:DesignWidth="640" d:DesignHeight="480"><Grid x:Name="LayoutRoot" Width="200" Margin="100" ><Grid.RowDefinitions><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="Auto"></RowDefinition><RowDefinition Height="*"></RowDefinition></Grid.RowDefinitions><Grid.ColumnDefinitions><ColumnDefinition Width="Auto" ></ColumnDefinition><ColumnDefinition Width="*" ></ColumnDefinition></Grid.ColumnDefinitions><TextBlock Grid.Row="0" Grid.Column="0" Margin="5">UserName:</TextBlock><TextBox Grid.Row="0" Grid.Column="1" Margin="5" Name="txtUserName" Text="admin" ></TextBox><TextBlock Grid.Row="1" Grid.Column="0" Margin="5">Password:</TextBlock><TextBox  Grid.Row="1" Grid.Column="1" Margin="5" Name="txtPassword" Text="123456"></TextBox><Button Name="btnLogin"  Grid.Row="2" Grid.ColumnSpan="2" Margin="5" Content="Login" Click="Login"></Button><Button Name="btnSignOut" Grid.Row="3" Grid.ColumnSpan="2" Margin="5" Content="SignOut" Click="SignOut"></Button><Button Name="btnSetWeather" Grid.Row="4" Grid.ColumnSpan="2" Margin="5" Content="SetWeather" Click="SetWeather"></Button><Button Name="btnGetWeather" Grid.Row="5" Grid.ColumnSpan="2" Margin="5" Content="GetWeather" Click="GetWeather"></Button></Grid>
</UserControl>

MainPage.xaml.cs文件:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;
using WcfSecSample.LoginServiceRef;
using WcfSecSample.WeatherServiceRef;namespace WcfSecSample
{public partial class MainPage : UserControl{public MainPage(){InitializeComponent();this.IsLogin = false;this.client.SetWeatherCompleted += new EventHandler<System.ComponentModel.AsyncCompletedEventArgs>(client_SetWeatherCompleted);this.client.GetWeatherCompleted += new EventHandler<GetWeatherCompletedEventArgs>(client_GetWeatherCompleted);this.loginClient.LoginCompleted += new EventHandler<LoginCompletedEventArgs>(loginClient_LoginCompleted);this.loginClient.SignOutCompleted += new EventHandler<System.ComponentModel.AsyncCompletedEventArgs>(loginClient_SignOutCompleted);}private bool isLogin;private WeatherServiceClient client = new WeatherServiceClient();private LoginServiceClient loginClient = new LoginServiceClient();private bool IsLogin{get { return isLogin; }set{isLogin = value;this.btnLogin.IsEnabled = !isLogin;this.btnSignOut.IsEnabled = isLogin;}}private void Login(object sender, RoutedEventArgs e){this.loginClient.LoginAsync(this.txtUserName.Text, this.txtPassword.Text);}private void SignOut(object sender, RoutedEventArgs e){this.loginClient.SignOutAsync();}private void GetWeather(object sender, RoutedEventArgs e){this.client.GetWeatherAsync();}private void SetWeather(object sender, RoutedEventArgs e){this.client.SetWeatherAsync("Cloudy");}void loginClient_LoginCompleted(object sender, LoginCompletedEventArgs e){if (e.Error == null){MessageBox.Show(e.Result ? "Login succeed." : "Login faild.");this.IsLogin = e.Result;}else{MessageBox.Show(e.Error.Message);}}void loginClient_SignOutCompleted(object sender, System.ComponentModel.AsyncCompletedEventArgs e){if (e.Error == null){MessageBox.Show("SignOut.");this.IsLogin = false;}else{MessageBox.Show(e.Error.Message);}}void client_GetWeatherCompleted(object sender, GetWeatherCompletedEventArgs e){if (e.Error == null) MessageBox.Show(e.Result);else MessageBox.Show(e.Error.Message);}void client_SetWeatherCompleted(object sender, System.ComponentModel.AsyncCompletedEventArgs e){if (e.Error == null){MessageBox.Show("Set weather succeed.");}else{MessageBox.Show(e.Error.Message);}}}
}

为了测试登录对访问服务的影响,以上代码并未根据登录状态对SetWeather、GetWeather按钮的可用性进行控制。

运行示例,可以看到在登录之前访问WeatherService是不成功的,如果用Admin角色的账号登录之后可以SetWeather或GetWeather,如果用Guest角色的账号登录则只能GetWeather。登录并调用GetWeather的效果图如下:

image

示例测试环境:

操作系统:Windows7

开发环境:Visual Studio 2008 + Silverlight 3

IIS:7.5

浏览器:IE8

转载于:https://www.cnblogs.com/chinadhf/archive/2010/04/29/1724388.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/406217.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javase基础复习攻略《七》

容器是什么&#xff1f;通俗的讲容器指可以装其它东西的器皿&#xff0c;前面我们提到的数组便是容器的一种&#xff0c;容器的概念在JAVA中便可以理解为用来存储其它对象的器皿。本篇就让我们一起来认识一下JAVA为我们提供的容器类。 1、容器API&#xff1a; J2SDK提供的容器A…

mysql2000数据库四合一_MSSQL2000四合一

MSSQL2000四合一是包含了MS Sql Server 2000 DEVELOPER、ENTERPRISE、PERSONAL、STANDARD 四个版本。在解压目录下有 SQL2000-KB884525-SP4-x86-CHS.EXE 这是SQL 2000 的 sp4 补丁。记录安装完后一定要安装上。个人版、企业版、专业版、开发版SQL Server 2000 是 Microsoft .NE…

jms mysql_JMS学习九(ActiveMQ的消息持久化到Mysql数据库)

1、将连接Mysql数据库的jar文件&#xff0c;放到ActiveMQ的lib目录下2、修改ActiveMQ的conf目录下的active.xml文件&#xff0c;修改数据持久化的方式2.1 修改原来的kshadb的持久化数据的方式2.2 连接Mysql的配置(注意配置文件放置的位置)3、将数据持久化Mysql的运行截图3.1 …

看完这篇不要告诉我不会封装ant design弹框组件了

我是歌谣 放弃很容易 但是坚持一定很酷 微信公众号关注前端小歌谣 获取前端学习知识 1设计需求 封装一个弹框组件 直接调用接口 2技术栈 ant designreact 设计第一步 绘制样式 <ModalmaskClosable{false}visible{visible}title{签收协议}onOk{this.handleSignFor}onCancel…

mysql 报错3534_win7下安装MYSQL报错:MYSQL 服务无法启动的3534问题

win7下安装MYSQL,只到“net start mysql”这一步报错:3534的错误&#xff1a;是直接官网下载的压缩文件.不是安装文件.解决方法:1:环境变量PATH添加完成(例如&#xff1a;h:\mysql\\bin2:在mysql目录下&#xff0c;新建data目录。3:在mysql目录下 新建新建一个默认配置文件my.i…

Impress.js上手 - 抛开PPT、制作Web 3D幻灯片放映

前言&#xff1a; 如果你已经厌倦了使用PPT设置路径、设置时间、设置动画方式来制作动画特效、那么Impress.js将是你一个非常好的选择。 用它制作的PPT将更加直观、效果也是嗷嗷美观的。 当然&#xff0c;如果用它来装X&#xff0c;是需要付出一些代价的&#xff0c;不过如果你…

mysql gtid 主主_mysql GTID主从复制(主库在线,添加新丛库)

要求&#xff1a;1、 主库上线&#xff0c;主库不停止服务的前提下做主从复制2、 新添加一个丛库操作&#xff1a;1、 在主库导出数据(主库正常运行)&#xff1b;2、 将主库的sql文件传到丛库&#xff1b;3、 丛库恢复数据库&#xff…

JS 画饼图,折线图

网址&#xff1a; http://www.hcharts.cn/demo/index.php 效果图&#xff1a; 它的网址里面都很全的。简单实用扣代码即可 使用时注意数据格式即可 1         //获取mood_evalue的百分比2 $total_mood_evalue 0;3 //初始化key的数组&…

vim、gvim在windows下中文乱码的终极解决方案

From: http://www.liuhuadong.com/archives/68/index.html vim、gvim在windows下中文乱码的终极解决方案 在windows下vim的中文字体显示并不好&#xff0c;所以我们需要调整字体显示编码为utf-8&#xff0c;只有这样才能使用美观的中文。这时候可能出现很多中文乱码。 如&#…

mysql业务繁忙时能建索引吗_MySQL DBA面试高频三十问

原标题&#xff1a;MySQL DBA面试高频三十问前言本文主要受众为开发人员,所以不涉及到MySQL的服务部署等操作,且内容较多,大家准备好耐心和瓜子矿泉水.前一阵系统的学习了一下MySQL,也有一些实际操作经验,偶然看到一篇和MySQL相关的面试文章,发现其中的一些问题自己也回答不好,…

Android小項目之---吃飯選哪?--》選擇對話框(附源碼)

還記得早先我們做的記算器的例子嗎&#xff1f;當中的驗證判斷用到了對話框&#xff0c;今天我們來做一個不一樣的對話框&#xff0c;要做的這個小例子是一個可供選擇效果的對話框 即層層迭迭的Alert Dialog&#xff1b;界面方面我們擺放一個Button來做一個按鈕事件&#xff0c…

UVa 11059 Maximum Product

题意&#xff1a;给出n个数组成的序列&#xff0c;求乘积最大的连续子序列 看的紫书&#xff0c;因为n最大为18&#xff0c;每个数最大为10,所以10^18用long long 能够存下&#xff0c; 直接枚举起点和终点找最大值就可以了 1 #include<iostream> 2 #include<cstdio&…

python 高维数据_用Sci-kit learn和XGBoost进行多类分类:Brainwave数据案例研究

在机器学习中&#xff0c;高维数据的分类问题非常具有挑战性。有时候&#xff0c;非常简单的问题会因为这个“维度诅咒”问题变得非常复杂。在本文中&#xff0c;我们将了解不同分类器的准确性和性能是如何变化的。理解数据对于本文&#xff0c;我们将使用Kaggle的“EEG Brainw…

缔造完美运维 共谱双赢新篇

国华电力早在许久之前就曾经使用北塔网络监控产品&#xff0c;但由于当前国华电力信息中心的网络管理系统呈分散状态&#xff0c;其范围也非常有限&#xff0c;致使无法全面了解国华电力客户信息网中计算机及网络系统的实时运行状况&#xff0c;对网络资源配置及网络流量分布也…