net/addr/ip: Add In4Addr FromStr conversion

Change-Id: I671e8c318aa03dcb1c12ce22d9ace43cdbe46dcd
Signed-off-by: William A. Kennington III <wak@google.com>
diff --git a/include/stdplus/net/addr/ip.hpp b/include/stdplus/net/addr/ip.hpp
index f21275f..0f5b30f 100644
--- a/include/stdplus/net/addr/ip.hpp
+++ b/include/stdplus/net/addr/ip.hpp
@@ -2,9 +2,13 @@
 #include <netinet/in.h>
 
 #include <stdplus/hash.hpp>
+#include <stdplus/numeric/endian.hpp>
+#include <stdplus/numeric/str.hpp>
+#include <stdplus/str/conv.hpp>
 #include <stdplus/variant.hpp>
 
 #include <algorithm>
+#include <stdexcept>
 #include <variant>
 
 namespace stdplus
@@ -49,6 +53,30 @@
     }
 };
 
+template <>
+struct FromStr<In4Addr>
+{
+    template <typename CharT>
+    constexpr In4Addr operator()(std::basic_string_view<CharT> sv) const
+    {
+        constexpr StrToInt<10, uint8_t> sti;
+        uint32_t addr = {};
+        for (size_t i = 0; i < 3; ++i)
+        {
+            auto loc = sv.find(".");
+            addr |= sti(sv.substr(0, loc));
+            addr <<= 8;
+            sv.remove_prefix(loc == sv.npos ? sv.size() : loc + 1);
+            if (sv.empty())
+            {
+                throw std::invalid_argument("Missing addr data");
+            }
+        }
+        addr |= sti(sv);
+        return in_addr{hton(addr)};
+    }
+};
+
 struct In6Addr : in6_addr
 {
     constexpr In6Addr() noexcept : in6_addr() {}
diff --git a/src/net/addr/ip.cpp b/src/net/addr/ip.cpp
index 466c509..81d2a4a 100644
--- a/src/net/addr/ip.cpp
+++ b/src/net/addr/ip.cpp
@@ -1 +1,6 @@
 #include <stdplus/net/addr/ip.hpp>
+
+namespace stdplus
+{
+template In4Addr FromStr<In4Addr>::operator()(std::string_view) const;
+}
diff --git a/test/net/addr/ip.cpp b/test/net/addr/ip.cpp
index 7d0cace..625633e 100644
--- a/test/net/addr/ip.cpp
+++ b/test/net/addr/ip.cpp
@@ -14,6 +14,22 @@
     std::hash<In4Addr>{}(In4Addr{});
 }
 
+TEST(FromStr, In4Addr)
+{
+    EXPECT_THROW(fromStr<In4Addr>(""), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("0.0.0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("0.0.0."), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>(".0.0.0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("0.0.0.0.0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("x.0.0.0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("ff.0.0.0"), std::invalid_argument);
+    EXPECT_THROW(fromStr<In4Addr>("256.0.0.0"), std::overflow_error);
+
+    EXPECT_EQ((In4Addr{}), fromStr<In4Addr>("0.0.0.0"));
+    EXPECT_EQ((In4Addr{192, 168, 1, 1}), fromStr<In4Addr>("192.168.001.1"));
+}
+
 TEST(EqualOperator, In6Addr)
 {
     EXPECT_EQ((In6Addr{0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff}),