《Java编码指南:编写安全可靠程序的75条建议》—— 指南21:不要让不可信代码误用回调方法的特权

简介: 回调提供一种注册方法的手段,在其感兴趣的事件发生时将会被调用(或回调。Java在很多地方都使用了回调, 如applet技术、响应Servlet的生命周期事件、AWT和Swing框架中的事件通知(如按钮单击事件),以及异步的读写操作等。

本节书摘来异步社区《Java编码指南:编写安全可靠程序的75条建议》一书中的第1章,第1.21节,作者:【美】Fred Long(弗雷德•朗), Dhruv Mohindra(德鲁•莫欣达), Robert C.Seacord(罗伯特 C.西科德), Dean F.Sutherland(迪恩 F.萨瑟兰), David Svoboda(大卫•斯沃博达),更多章节内容可以访问云栖社区“异步社区”公众号查看。

指南21:不要让不可信代码误用回调方法的特权

回调提供一种注册方法的手段,在其感兴趣的事件发生时将会被调用(或回调。Java在很多地方都使用了回调, 如applet技术、响应Servlet的生命周期事件、AWT和Swing框架中的事件通知(如按钮单击事件),以及异步的读写操作等。甚至在线程的运行机制Runnable.run()中,新起一个线程时,自动执行对应的run()方法都使用到了回调技术。

在Java中,回调函数通常是使用接口来实现。下面是回调的一般结构:

public interface CallBack {
 void callMethod();
}

class CallBackImpl implements CallBack {
 public void callMethod() {
  System.out.println("CallBack invoked");
 }
}

class CallBackAction {
 private CallBack callback;
 public CallBackAction(CallBack callback) {
  this.callback = callback;
 }

 public void perform() {
  callback.callMethod();
 }
}

class Client {
 public static void main(String[] args) {
  CallBackAction action =
   new CallBackAction(new CallBackImpl());
  // ...
  action.perform(); // Prints "CallBack invoked"
 }
}```
回调方法的调用通常没有特权的变化,这意味着,执行它们的上下文的特权可能比声明它们的上下文的多。如果这些回调方法接受不可信代码的数据,那么就有可能发生特权升级。

根据Oracle的安全编码指南[SCG 2010]:

从系统调用回调方法通常具有完全权限。恶意代码在执行操作时会出现在栈上——这似乎是一个合理的期望,但事实并非如此。恶意代码可以设置一个对象,用来将回调过渡给一个经过安全检查的操作。例如,一个文件选择器对话框,可以从用户动作中操作文件系统,这就可能导致恶意代码发送某些事件。另外,恶意代码可以通过将看上去无害的东西伪装成一个文件选择器来重定向用户事件。

这条指南是指南17的一个实例,并且和《The CERT® Oracle® Secure Coding Standard for Java™》[Long 2012]的“SEC01-J.Do not allow tainted variables in privileged blocks”有关。

####违规代码示例
下面的违规代码示例使用UserLookupCallBack类实现CallBack接口,通过给定的用户ID查找用户名。这个查找代码假定这些信息在/etc/passwd文件中,这需要提升特权才能打开。因此,Client类使用提升的特权(在doPrivileged语句块中)来调用所有回调。

public interface CallBack {
 void callMethod();
}

class UserLookupCallBack implements CallBack {
 private int uid;
 private String name;
 public UserLookupCallBack(int uid) {
  this.uid = uid;
 }

 public String getName() {
  return name;
 }

 public void callMethod() {
  try (InputStream fis = new FileInputStream("/etc/passwd")) {
   // Look up uid & assign to name
  } catch (IOException x) {
   name = null;
  }
 }
}

final class CallBackAction {
 private CallBack callback;

 public CallBackAction(CallBack callback) {
  this.callback = callback;
 }

 public void perform() {
  AccessController.doPrivileged(new PrivilegedAction() {
   public Void run() {
    callback.callMethod();
    return null;
   }
  });
 }
}`
这段代码可以被客户端安全地使用,如下所示:

public static void main(String[] args) {
 int uid = Integer.parseInt(args[0]);
 CallBack callBack = new UserLookupCallBack(uid);
 CallBackAction action = new CallBackAction(callBack);

 // ...
 action.perform(); // Looks up user name
 System.out.println("User " + uid + " is named " +
           callBack.getName());
}```
然而,攻击者可以通过注册MaliciousCallBack实例,使用CallBackAction和提升的特权执行恶意代码:

class MaliciousCallBack implements CallBack {
 public void callMethod() {
  // Code here gets executed with elevated privileges
 }
}
// Client code
public static void main(String[] args) {
 CallBack callBack = new MaliciousCallBack();
 CallBackAction action = new CallBackAction(callBack);
 action.perform(); // Executes malicious code
}`
合规解决方案(回调自己调用doPrivileged块)
根据Oracle公司的安全编码指南[SCG 2010]:

按照惯例,PrivilegedAction的实例和PrivilegedExceptionAction的实例可以提供给不可信代码,但不能以调用者提供的动作调用doPrivileged。

下面的合规解决方案将doPrivileged()的调用从CallBackAction代码中移到了它自己的回调里。

public interface CallBack {
 void callMethod();
}

class UserLookupCallBack implements CallBack {
 private int uid;
 private String name;
 public UserLookupCallBack(int uid) {
  this.uid = uid;
 }

 public String getName() {
  return name;
 }

 public final void callMethod() {
  AccessController.doPrivileged(new PrivilegedAction<Void>() {
   public Void run() {
    try (InputStream fis =
     new FileInputStream("/etc/passwd")) {
     // Look up userid and assign to
     // UserLookupCallBack.this.name
    } catch (IOException x) {
     UserLookupCallBack.this.name = null;
    }
    return null;
   }
  });
 }
}

final class CallBackAction {
 private CallBack callback;

 public CallBackAction(CallBack callback) {
  this.callback = callback;
 }

 public void perform() {
  callback.callMethod();
 }
}```
这段代码的行为和之前一样,但攻击者不能再以提升的特权执行恶意回调代码。即使攻击者可以通过使用CallBackAction类的构造函数传递一个恶意的回调实例,代码也不能以提升的特权执行,因为恶意实例必须包含的doPrivileged语句块没有和可信代码一样的权限。此外,因为CallBackAction类已被声明为final,所以它不能被子类化,从而不能覆盖perform()方法。

合规解决方案(将回调声明为final)
下面的合规解决方案通过将UserLookupCallBack类声明为final来防止callMethod()被覆盖。

final class UserLookupCallBack implements CallBack {
 // ...
}

// Remaining code is unchanged`

适用性

通过回调暴露敏感方法可能导致特权误用和任意代码执行。

相关文章
|
1天前
|
自然语言处理 Java
Java中的字符集编码入门-增补字符(转载)
本文探讨Java对Unicode的支持及其发展历程。文章详细解析了Unicode字符集的结构,包括基本多语言面(BMP)和增补字符的表示方法,以及UTF-16编码中surrogate pair的使用。同时介绍了代码点和代码单元的概念,并解释了UTF-8的编码规则及其兼容性。
73 60
|
10天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
33 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
53 24
|
18天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
54 2
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
76 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
55 5
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
2月前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
27 1
|
1月前
|
安全 Java API
Java中的Lambda表达式:简化代码的现代魔法
在Java 8的发布中,Lambda表达式的引入无疑是一场编程范式的革命。它不仅让代码变得更加简洁,还使得函数式编程在Java中成为可能。本文将深入探讨Lambda表达式如何改变我们编写和维护Java代码的方式,以及它是如何提升我们编码效率的。
|
2月前
|
SQL 安全 Java
Java 异常处理:筑牢程序稳定性的 “安全网”
本文深入探讨Java异常处理,涵盖异常的基础分类、处理机制及最佳实践。从`Error`与`Exception`的区分,到`try-catch-finally`和`throws`的运用,再到自定义异常的设计,全面解析如何有效管理程序中的异常情况,提升代码的健壮性和可维护性。通过实例代码,帮助开发者掌握异常处理技巧,确保程序稳定运行。
46 0