package nokogiri; import static nokogiri.XmlNode.setDocumentAndDecorate; import static nokogiri.internals.NokogiriHelpers.getNokogiriClass; import static nokogiri.internals.NokogiriHelpers.nodeListToRubyArray; import java.util.Arrays; import org.jruby.Ruby; import org.jruby.RubyArray; import org.jruby.RubyClass; import org.jruby.RubyFixnum; import org.jruby.RubyObject; import org.jruby.RubyRange; import org.jruby.anno.JRubyClass; import org.jruby.anno.JRubyMethod; import org.jruby.runtime.ThreadContext; import org.jruby.runtime.Visibility; import org.jruby.runtime.builtin.IRubyObject; import org.w3c.dom.Node; import org.w3c.dom.NodeList; /** * Class for Nokogiri::XML::NodeSet * * @author sergio * @author Yoko Harada */ @JRubyClass(name = "Nokogiri::XML::NodeSet") public class XmlNodeSet extends RubyObject implements NodeList { private static final long serialVersionUID = 1L; IRubyObject[] nodes; public XmlNodeSet(Ruby ruby, RubyClass klazz) { super(ruby, klazz); nodes = IRubyObject.NULL_ARRAY; } XmlNodeSet(Ruby ruby, RubyClass klazz, IRubyObject[] nodes) { super(ruby, klazz); this.nodes = nodes; } public static XmlNodeSet newEmptyNodeSet(ThreadContext context, XmlNodeSet docOwner) { final Ruby runtime = context.runtime; XmlNodeSet set = new XmlNodeSet(runtime, getNokogiriClass(runtime, "Nokogiri::XML::NodeSet")); set.initializeFrom(context, docOwner); return set; } public static XmlNodeSet newEmptyNodeSet(ThreadContext context, XmlNode docOwner) { final Ruby runtime = context.runtime; XmlNodeSet set = new XmlNodeSet(runtime, getNokogiriClass(runtime, "Nokogiri::XML::NodeSet")); set.initialize(runtime, docOwner); return set; } public static XmlNodeSet newNodeSet(Ruby runtime, IRubyObject[] nodes) { XmlNodeSet xmlNodeSet = new XmlNodeSet(runtime, getNokogiriClass(runtime, "Nokogiri::XML::NodeSet")); xmlNodeSet.setNodes(nodes); return xmlNodeSet; } public static XmlNodeSet newNodeSet(Ruby runtime, IRubyObject[] nodes, XmlNode docOwner) { XmlNodeSet set = new XmlNodeSet(runtime, getNokogiriClass(runtime, "Nokogiri::XML::NodeSet"), nodes); set.initialize(runtime, docOwner); return set; } /** * Create and return a copy of this object. * * @return a clone of this object */ @Override public Object clone() throws CloneNotSupportedException { return super.clone(); } private void setNodes(IRubyObject[] array) { this.nodes = array; IRubyObject first = array.length > 0 ? array[0] : null; initialize(getRuntime(), first); } private void initializeFrom(ThreadContext context, XmlNodeSet ref) { IRubyObject document = ref.getInstanceVariable("@document"); if (document != null && !document.isNil()) { initialize(context, (XmlDocument) document); } } final void initialize(Ruby runtime, IRubyObject refNode) { if (refNode instanceof XmlNode) { XmlDocument doc = ((XmlNode) refNode).document(runtime); setDocumentAndDecorate(runtime.getCurrentContext(), this, doc); } } private void initialize(ThreadContext context, XmlDocument doc) { setDocumentAndDecorate(context, this, doc); } public int length() { return nodes == null ? 0 : nodes.length; } @JRubyMethod(name = "&") public IRubyObject op_and(ThreadContext context, IRubyObject nodeSet) { IRubyObject[] otherNodes = getNodes(context, nodeSet); if (otherNodes == null || otherNodes.length == 0) { return newEmptyNodeSet(context, this); } if (nodes == null || nodes.length == 0) { return newEmptyNodeSet(context, this); } IRubyObject[] curr = nodes; IRubyObject[] other = getNodes(context, nodeSet); IRubyObject[] result = new IRubyObject[nodes.length]; int last = 0; outer: for (int i = 0; i < curr.length; i++) { IRubyObject n = curr[i]; for (int j = 0; j < other.length; j++) { if (other[j] == n) { result[last++] = n; continue outer; } } } XmlNodeSet newSet = newNodeSet(context.runtime, Arrays.copyOf(result, last)); newSet.initializeFrom(context, this); return newSet; } @JRubyMethod public IRubyObject delete (ThreadContext context, IRubyObject node_or_namespace) { IRubyObject nodeOrNamespace = asXmlNodeOrNamespace(context, node_or_namespace); if (nodes.length == 0) { return context.nil; } IRubyObject[] orig = nodes; IRubyObject[] result = new IRubyObject[nodes.length]; int last = 0; for (int i = 0; i < orig.length; i++) { IRubyObject n = orig[i]; if (n == nodeOrNamespace) { continue; } result[last++] = n; } nodes = Arrays.copyOf(result, last); if (nodes.length < orig.length) { // if we found the node return it return nodeOrNamespace; } return context.nil; } public IRubyObject dup(ThreadContext context) { XmlNodeSet dup = newNodeSet(context.runtime, nodes.clone()); dup.initializeFrom(context, this); return dup; } @JRubyMethod(visibility = Visibility.PROTECTED) public IRubyObject initialize_copy(ThreadContext context, IRubyObject other) { setNodes(getNodes(context, other)); initializeFrom(context, (XmlNodeSet)other); return this; } @JRubyMethod(name = "include?") public IRubyObject include_p(ThreadContext context, IRubyObject node_or_namespace) { for (int i = 0; i < nodes.length; i++) { if (nodes[i] == node_or_namespace) { return context.tru; } } return context.fals; } @JRubyMethod(name = {"length", "size"}) public IRubyObject length(ThreadContext context) { return context.runtime.newFixnum(nodes.length); } @JRubyMethod(name = "-") public IRubyObject op_diff(ThreadContext context, IRubyObject nodeSet) { IRubyObject[] otherNodes = getNodes(context, nodeSet); if (otherNodes.length == 0) { return dup(context); } if (nodes.length == 0) { return newEmptyNodeSet(context, this); } IRubyObject[] curr = nodes; IRubyObject[] other = getNodes(context, nodeSet); IRubyObject[] result = new IRubyObject[nodes.length]; int last = 0; outer: for (int i = 0; i < curr.length; i++) { IRubyObject n = curr[i]; for (int j = 0; j < other.length; j++) { if (other[j] == n) { continue outer; } } result[last++] = n; } XmlNodeSet newSet = newNodeSet(context.runtime, Arrays.copyOf(result, last)); newSet.initializeFrom(context, this); return newSet; } @JRubyMethod(name = {"|", "+"}) public IRubyObject op_or(ThreadContext context, IRubyObject nodeSet) { IRubyObject[] otherNodes = getNodes(context, nodeSet); if (nodes.length == 0) { return ((XmlNodeSet) nodeSet).dup(context); } if (otherNodes.length == 0) { return dup(context); } IRubyObject[] curr = nodes; IRubyObject[] other = getNodes(context, nodeSet); IRubyObject[] result = Arrays.copyOf(curr, curr.length + other.length); int last = curr.length; outer: for (int i = 0; i < other.length; i++) { IRubyObject n = other[i]; for (int j = 0; j < curr.length; j++) { if (curr[j] == n) { continue outer; } } result[last++] = n; } XmlNodeSet newSet = newNodeSet(context.runtime, Arrays.copyOf(result, last)); newSet.initializeFrom(context, this); return newSet; } @JRubyMethod(name = {"push", "<<"}) public IRubyObject push(ThreadContext context, IRubyObject node_or_namespace) { nodes = Arrays.copyOf(nodes, nodes.length + 1); nodes[nodes.length - 1] = node_or_namespace; return this; } // replace with // https://github.com/jruby/jruby/blame/13a3ec76d883a162b9d46c374c6e9eeea27b3261/core/src/main/java/org/jruby/RubyRange.java#L974 // once we upgraded the min JRuby version to >= 9.2 private static IRubyObject rangeBeginLength(ThreadContext context, IRubyObject rangeMaybe, int len, int[] begLen) { RubyRange range = (RubyRange) rangeMaybe; int min = range.begin(context).convertToInteger().getIntValue(); int max = range.end(context).convertToInteger().getIntValue(); if (min < 0) { min += len; if (min < 0) { throw context.runtime.newRangeError(min + ".." + (range.isExcludeEnd() ? "." : "") + max + " out of range"); } } if (max < 0) { max += len; } if (!range.isExcludeEnd()) { max++; } begLen[0] = min; begLen[1] = max; return context.tru; } @JRubyMethod(name = {"[]", "slice"}) public IRubyObject slice(ThreadContext context, IRubyObject indexOrRange) { if (indexOrRange instanceof RubyFixnum) { return slice(context, ((RubyFixnum) indexOrRange).getIntValue()); } if (indexOrRange instanceof RubyRange) { int[] begLen = new int[2]; rangeBeginLength(context, indexOrRange, nodes.length, begLen); int min = begLen[0]; int max = begLen[1]; return subseq(context, min, max - min); } throw context.runtime.newTypeError("index must be an Integer or a Range"); } IRubyObject slice(ThreadContext context, int idx) { if (idx < 0) { idx += nodes.length; } if (idx >= nodes.length || idx < 0) { return context.nil; } return nodes[idx]; } @JRubyMethod(name = {"[]", "slice"}) public IRubyObject slice(ThreadContext context, IRubyObject start, IRubyObject length) { int s = ((RubyFixnum) start).getIntValue(); int l = ((RubyFixnum) length).getIntValue(); if (s < 0) { s += nodes.length; } return subseq(context, s, l); } public IRubyObject subseq(ThreadContext context, int start, int length) { if (start > nodes.length) { return context.nil; } if (start < 0 || length < 0) { return context.nil; } if (start + length > nodes.length) { length = nodes.length - start; } int to = start + length; return newNodeSet(context.runtime, Arrays.copyOfRange(nodes, start, to)); } @JRubyMethod(name = {"to_a", "to_ary"}) public RubyArray to_a(ThreadContext context) { return context.runtime.newArrayNoCopy(nodes); } @JRubyMethod(name = {"unlink", "remove"}) public IRubyObject unlink(ThreadContext context) { for (int i = 0; i < nodes.length; i++) { if (nodes[i] instanceof XmlNode) { ((XmlNode) nodes[i]).unlink(context); } } return this; } private static IRubyObject asXmlNodeOrNamespace(ThreadContext context, IRubyObject possibleNode) { if (possibleNode instanceof XmlNode || possibleNode instanceof XmlNamespace) { return possibleNode; } throw context.getRuntime().newArgumentError("node must be a Nokogiri::XML::Node or Nokogiri::XML::Namespace"); } private static IRubyObject[] getNodes(ThreadContext context, IRubyObject possibleNodeSet) { if (possibleNodeSet instanceof XmlNodeSet) { return ((XmlNodeSet) possibleNodeSet).nodes; } throw context.getRuntime().newArgumentError("node must be a Nokogiri::XML::NodeSet"); } public int getLength() { return nodes.length; } public Node item(int index) { Object n = nodes[index]; if (n instanceof XmlNode) { return ((XmlNode) n).node; } if (n instanceof XmlNamespace) { return ((XmlNamespace) n).getNode(); } return null; } }