c# - 在参数值中禁用等号的 URL 编码
问题描述
我在下面有这段代码,它为一个安静的 Web 服务调用创建查询参数集合
NameValueCollection outgoingQueryString = HttpUtility.ParseQueryString(String.Empty);
outgoingQueryString.Add("SessionID", _sessionId);
outgoingQueryString.Add("LoanNumberID", loanId.ToString());
outgoingQueryString.Add("Nonce", _nonce);
outgoingQueryString.Add("OtherParams", "Enable_Filter_Show_on_Action_Report=0");
问题是最后一个参数值 Enable_Filter_Show_on_Action_Report=0 得到它的 '=' 编码为 %3dSessionID=319A561B6D&LoanNumberID=351591&Nonce=3262383361&OtherParams=Enable_Filter_Show_on_Action_Report%3d0
什么时候应该
SessionID=319A561B6D&LoanNumberID=351591&Nonce=3262383361&OtherParams=Enable_Filter_Show_on_Action_Report=0
事后没有进行重写,有没有办法防止这种情况发生?该服务未返回正确的结果集,因为它将 %3d 解释为需要过滤的内容。
我为解决此问题所做的工作如下。我只是不知道是否有办法首先防止这种情况。
string queryString = outgoingQueryString.ToString().Replace("%3d", "=");
解决方案
我不知道NameValueCollection
如何去做你想做的事;您可能会遇到某种解决方法。
我确实在您的解决方法实施中看到了一个潜在的问题:
[Fact]
public void ProblemWithOriginalSolution()
{
NameValueCollection outgoingQueryString = HttpUtility.ParseQueryString(String.Empty);
outgoingQueryString.Add("Param1", "Value1");
outgoingQueryString.Add("Param2", "Value2=Something"); // this '=' needs to remain URL-encoded
outgoingQueryString.Add("OtherParams", "Enable_Filter_Show_on_Action_Report=0");
var queryString = outgoingQueryString.ToString().Replace("%3d", "=");
Assert.Contains("Value2=Something", queryString); // Passes, but not what we want
Assert.Contains("Value2%3dSomething", queryString); // Actually what we want, but fails
}
这对您的需求可能无关紧要,但如果确实如此,我有两个建议。
第一个是只使用字符串连接。根据您的使用,这可能需要额外的逻辑来处理OtherParams
唯一查询字符串参数的情况(在这种情况下应该省略与号)。字符串连接也是一个坏主意,OtherParams
因为它可能包含需要进行 URL 编码的字符。
[Fact]
public void Concatentation()
{
NameValueCollection outgoingQueryString = HttpUtility.ParseQueryString(String.Empty);
outgoingQueryString.Add("Param1", "Value1");
var queryString = outgoingQueryString.ToString() + "&OtherParams=Enable_Filter_Show_on_Action_Report=0";
Assert.Contains("&OtherParams=Enable_Filter_Show_on_Action_Report=0", queryString);
}
二是做双换人。这有点笨拙,效率不高,但应该是可靠的。
[Fact]
public void DoubleSubstitution()
{
// Make sure the placeholder doesn't have any characters that will get URL encoded!
// Also make sure it's not something that could appear in your input.
const string equalsPlaceholder = "__Equals__";
NameValueCollection outgoingQueryString = HttpUtility.ParseQueryString(String.Empty);
outgoingQueryString.Add("Param1", "Value1");
// First, remove the equals signs from the OtherParams value
outgoingQueryString.Add("OtherParams", "Enable_Filter_Show_on_Action_Report=0".Replace("=", equalsPlaceholder));
// Then, build the query string, and substitute the equals signs back in
var queryString = outgoingQueryString.ToString().Replace(equalsPlaceholder, "=");
Assert.Contains("&OtherParams=Enable_Filter_Show_on_Action_Report=0", queryString);
}
推荐阅读
- puma - Puma 服务器无法启动,libssl 错误 - 找不到文件
- javascript - 如何在 gnome 3.28 Ubuntu 18.04 中的 javascript 中获取用户个人资料图像
- r - 过滤同一行中的相同值 将相同的值加在一起
- sql - 为什么 DATEDIFF 将这些日期解析为 dd/mm/yyyy?
- python - 为什么我在使用 pipenv 安装 scrapy 时会出现此错误?
- javascript - 在 React 中过滤特定的表行
- mongodb - Mongoose 使用 Express 填充,不在生产环境中工作(Heroku)
- java - 有没有更好的方法可以在不使用反射的情况下从对象中克隆值?
- azure - Azcopy 在 Linux Openwrt 上失败(错误的函数名称)
- git - 如何从 VPN 中排除 GIT