Mercurial Hosting > luan
changeset 1387:bc40bc9aab3a
start postgres backup
author | Franklin Schmidt <fschmidt@gmail.com> |
---|---|
date | Mon, 02 Sep 2019 22:23:12 -0600 (2019-09-03) |
parents | dc36dd8bf839 |
children | 2024d23ddd64 |
files | examples/blog/src/delete.luan examples/blog/src/index.html.luan examples/blog/src/lib/Db.luan examples/blog/src/lib/Post.luan lib/postgresql-9.1-901.jdbc4.jar src/luan/modules/lucene/Lucene.luan src/luan/modules/lucene/LuceneIndex.java src/luan/modules/lucene/PostgresBackup.java |
diffstat | 8 files changed, 315 insertions(+), 8 deletions(-) [+] |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/examples/blog/src/delete.luan Mon Sep 02 22:23:12 2019 -0600 @@ -0,0 +1,13 @@ +local Luan = require "luan:Luan.luan" +local error = Luan.error +local String = require "luan:String.luan" +local to_number = String.to_number or error() +local Http = require "luan:http/Http.luan" +local Post = require "site:/lib/Post.luan" + + +return function() + local post_id = to_number(Http.request.parameters.post) or error() + Post.delete_by_id(post_id) + Http.response.send_redirect("/") +end
--- a/examples/blog/src/index.html.luan Thu Aug 22 12:32:20 2019 -0600 +++ b/examples/blog/src/index.html.luan Mon Sep 02 22:23:12 2019 -0600 @@ -39,7 +39,11 @@ %> <a name="p<%= post.id %>"> <h2><%= post.subject %></h2> - <p><%= Time.format(post.date) %> - <a href="edit.html?post=<%= post.id %>">Edit</a></p> + <p> + <%= Time.format(post.date) %> + - <a href="edit.html?post=<%= post.id %>">Edit</a> + - <a href="delete?post=<%= post.id %>">Delete</a> + </p> <pre><%= bbcode_to_html(html_encode(post.content)) %></pre> <hr> <%
--- a/examples/blog/src/lib/Db.luan Thu Aug 22 12:32:20 2019 -0600 +++ b/examples/blog/src/lib/Db.luan Mon Sep 02 22:23:12 2019 -0600 @@ -4,9 +4,13 @@ local Db = {} +local function completer(doc) + return doc +end + function Db.new(lucene_dir) local dir = Io.uri(lucene_dir) - local db = Lucene.index( dir, Lucene.type.english, {"subject","content"} ) + local db = Lucene.index( dir, Lucene.type.english, {"subject","content"}, completer ) -- this is how you index a field -- db.indexed_fields.post_date = Lucene.type.long
--- a/examples/blog/src/lib/Post.luan Thu Aug 22 12:32:20 2019 -0600 +++ b/examples/blog/src/lib/Post.luan Mon Sep 02 22:23:12 2019 -0600 @@ -67,4 +67,8 @@ return posts end +function Post.delete_by_id(id) + db.delete("id:"..id) +end + return Post
--- a/src/luan/modules/lucene/Lucene.luan Thu Aug 22 12:32:20 2019 -0600 +++ b/src/luan/modules/lucene/Lucene.luan Mon Sep 02 22:23:12 2019 -0600 @@ -35,12 +35,12 @@ Lucene.literal = SaneQueryParser.literal -function Lucene.index(index_dir,default_type,default_fields) +function Lucene.index(index_dir,default_type,default_fields,postgres_backup) type(index_dir)=="table" or error "index_dir must be table" index_dir.to_uri_string and matches(index_dir.to_uri_string(),"^file:") or error "must be file" local index = {} index.dir = index_dir - local java_index, closer = LuceneIndex.getLuceneIndex(index_dir.java.file,default_type,default_fields) + local java_index, closer = LuceneIndex.getLuceneIndex(index_dir.java.file,default_type,default_fields,postgres_backup) index.java = java_index index.closer = closer or error() @@ -69,6 +69,9 @@ index.count_tokens = java_index.count_tokens index.close = closer.close + index.has_postgres_backup = java_index.hasPostgresBackup() + index.rebuild_postgres_backup = java_index.rebuild_postgres_backup + function index.search( query, from, to, options ) from or error "missing 'from' parameter" to or error "missing 'to' parameter"
--- a/src/luan/modules/lucene/LuceneIndex.java Thu Aug 22 12:32:20 2019 -0600 +++ b/src/luan/modules/lucene/LuceneIndex.java Mon Sep 02 22:23:12 2019 -0600 @@ -45,6 +45,7 @@ import org.apache.lucene.util.BytesRef; import org.apache.lucene.util.NumericUtils; import org.apache.lucene.search.Query; +import org.apache.lucene.search.PrefixQuery; import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TopDocs; import org.apache.lucene.search.Sort; @@ -110,14 +111,14 @@ private static Map<String,LuceneIndex> indexes = new HashMap<String,LuceneIndex>(); - public static Object[] getLuceneIndex(Luan luan,File indexDir,FieldParser defaultFieldParser,String[] defaultFields) + public static Object[] getLuceneIndex(Luan luan,File indexDir,FieldParser defaultFieldParser,String[] defaultFields,LuanFunction completer) throws LuanException, IOException { String key = indexDir.getCanonicalPath(); synchronized(indexes) { LuceneIndex li = indexes.get(key); if( li == null ) { - li = new LuceneIndex(indexDir,defaultFieldParser,defaultFields,key); + li = new LuceneIndex(indexDir,defaultFieldParser,defaultFields,key,completer); li.openCount = 1; indexes.put(key,li); } else { @@ -157,7 +158,9 @@ private final FieldParser defaultFieldParser; private final String[] defaultFields; - private LuceneIndex(File indexDir,FieldParser defaultFieldParser,String[] defaultFields,String key) + private final PostgresBackup postgresBackup; + + private LuceneIndex(File indexDir,FieldParser defaultFieldParser,String[] defaultFields,String key,LuanFunction completer) throws LuanException, IOException { this.key = key; @@ -174,6 +177,9 @@ } this.analyzer = analyzer; reopen(); + postgresBackup = completer!=null ? PostgresBackup.newInstance() : null; + if( postgresBackup != null && postgresBackup.wasCreated ) + rebuild_postgres_backup(completer); } public void reopen() throws LuanException, IOException { @@ -199,6 +205,8 @@ try { writer.deleteAll(); id = idLim = 0; + if( postgresBackup != null ) + postgresBackup.deleteAll(); if(commit) writer.commit(); } finally { wrote(); @@ -212,6 +220,28 @@ return new Term(key,br); } + private void backupDelete(Query query) + throws IOException + { + if( postgresBackup != null ) { + final List<Long> ids = new ArrayList<Long>(); + IndexSearcher searcher = openSearcher(); + MyCollector col = new MyCollector() { + @Override public void collect(int iDoc) throws IOException { + Document doc = searcher.doc( docBase + iDoc ); + Long id = (Long)doc.getField("id").numericValue(); + ids.add(id); + } + }; + searcher.search(query,col); + postgresBackup.begin(); + for( Long id : ids ) { + postgresBackup.delete(id); + } + postgresBackup.commit(); + } + } + public void delete(String queryStr) throws IOException, ParseException { @@ -220,6 +250,7 @@ boolean commit = !writeLock.isHeldByCurrentThread(); writeLock.lock(); try { + backupDelete(query); writer.deleteDocuments(query); if(commit) writer.commit(); } finally { @@ -235,6 +266,9 @@ public void save(LuanTable doc,LuanTable boosts) throws LuanException, IOException { + if( boosts!=null && postgresBackup!=null ) + logger.error("boosts are not saved to postgres backup"); + Object obj = doc.get("id"); Long id; try { @@ -250,8 +284,12 @@ id = nextId(); doc.put("id",id); writer.addDocument(toLucene(doc,boosts)); + if( postgresBackup != null ) + postgresBackup.add(id,doc); } else { writer.updateDocument( term("id",id), toLucene(doc,boosts) ); + if( postgresBackup != null ) + postgresBackup.update(id,doc); } if(commit) writer.commit(); } finally { @@ -263,16 +301,31 @@ public Object run_in_transaction(LuanFunction fn) throws IOException, LuanException { boolean commit = !writeLock.isHeldByCurrentThread(); writeLock.lock(); + boolean ok = false; try { + if( commit && postgresBackup != null ) + postgresBackup.begin(); Object rtn = fn.call(); - if(commit) writer.commit(); + ok = true; + if(commit) { + if( postgresBackup != null ) + postgresBackup.commit(); + writer.commit(); + } return rtn; } finally { + if( !ok && commit ) { + if( postgresBackup != null ) + postgresBackup.rollback(); + writer.rollback(); + reopen(); + } wrote(); writeLock.unlock(); } } + // ??? public Object run_in_lock(LuanFunction fn) throws IOException, LuanException { if( writeLock.isHeldByCurrentThread() ) throw new RuntimeException(); @@ -373,6 +426,8 @@ } public void doClose() throws IOException { + if( postgresBackup != null ) + postgresBackup.close(); writer.close(); reader.close(); } @@ -725,4 +780,48 @@ return n; } + + + public boolean hasPostgresBackup() { + return postgresBackup != null; + } + + public void rebuild_postgres_backup(LuanFunction completer) + throws IOException, LuanException + { + writeLock.lock(); + boolean ok = false; + try { + postgresBackup.begin(); + postgresBackup.deleteAll(); + Query query = new PrefixQuery(new Term("id")); + IndexSearcher searcher = openSearcher(); + MyCollector col = new MyCollector() { + @Override public void collect(int iDoc) throws IOException { + try { + Document doc = searcher.doc( docBase + iDoc ); + LuanTable tbl = toTable(completer.luan(),doc); + tbl = (LuanTable)completer.call(tbl); + Long id = (Long)tbl.get("id"); + //logger.info("id = "+id); + postgresBackup.add(id,tbl); + } catch(LuanException e) { + throw new LuanRuntimeException(e); + } + } + }; + try { + searcher.search(query,col); + } catch(LuanRuntimeException e) { + throw (LuanException)e.getCause(); + } + ok = true; + postgresBackup.commit(); + } finally { + if( !ok ) + postgresBackup.rollback(); + writeLock.unlock(); + } + } + }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/luan/modules/lucene/PostgresBackup.java Mon Sep 02 22:23:12 2019 -0600 @@ -0,0 +1,180 @@ +package luan.modules.lucene; + +import java.sql.Connection; +import java.sql.DriverManager; +import java.sql.Statement; +import java.sql.PreparedStatement; +import java.sql.SQLException; +import java.util.Properties; +import luan.LuanTable; +import luan.LuanException; +import luan.modules.parsers.LuanToString; +import luan.lib.logging.Logger; +import luan.lib.logging.LoggerFactory; + + +final class PostgresBackup { + private static final Logger logger = LoggerFactory.getLogger(PostgresBackup.class); + + static PostgresBackup newInstance() { + try { + return new PostgresBackup(); + } catch(ClassNotFoundException e) { + logger.error("creation failed",e); + return null; + } catch(SQLException e) { + logger.error("creation failed",e); + return null; + } + } + + final boolean wasCreated; + private final Connection con; + private final PreparedStatement insertStmt; + private final PreparedStatement updateStmt; + private final PreparedStatement deleteStmt; + private int trans = 0; + + private PostgresBackup() + throws ClassNotFoundException, SQLException + { + Class.forName("org.postgresql.Driver"); + + String url = "jdbc:postgresql://localhost:5432/luan"; + Properties props = new Properties(); + props.setProperty("user","postgres"); + props.setProperty("password",""); + + con = DriverManager.getConnection(url,props); + + Statement stmt = con.createStatement(); + boolean hasTable = stmt.executeQuery( + "select * from information_schema.tables where table_name='lucene'" + ).next(); + if( !hasTable ) { + stmt.executeUpdate( + "create table lucene (" + +" id integer not null primary key," + +" data text not null" + +")" + ); + } + stmt.close(); + wasCreated = !hasTable; + + insertStmt = con.prepareStatement( + "insert into lucene (id,data) values (?,?)" + ); + updateStmt = con.prepareStatement( + "update lucene set data=? where id=?" + ); + deleteStmt = con.prepareStatement( + "delete from lucene where id=?" + ); + } + + void close() { + try { + insertStmt.close(); + updateStmt.close(); + deleteStmt.close(); + con.close(); + } catch(SQLException e) { + logger.error("close failed",e); + } + } + + protected void finalize() throws Throwable { + super.finalize(); + if( !con.isClosed() ) { + logger.error("con not closed"); + con.close(); + } + } + + void add(long id,LuanTable doc) throws LuanException { + try { +//logger.info("getAutoCommit="+con.getAutoCommit()); + String data = LuanToString.toString(doc,true); + insertStmt.setLong(1,id); + insertStmt.setString(2,data); + insertStmt.executeUpdate(); + } catch(SQLException e) { + logger.error("add failed",e); + } + } + + void update(long id,LuanTable doc) throws LuanException { + try { + String data = LuanToString.toString(doc,true); + updateStmt.setString(1,data); + updateStmt.setLong(2,id); + int n = updateStmt.executeUpdate(); + if( n==0 ) { + logger.error("update not found for id="+id+", trying add"); + add(id,doc); + } else if( n!=1 ) + throw new RuntimeException(); + } catch(SQLException e) { + logger.error("update failed",e); + } + } + + void deleteAll() { + try { + Statement stmt = con.createStatement(); + stmt.executeUpdate("delete from lucene"); + stmt.close(); + } catch(SQLException e) { + logger.error("update failed",e); + } + } + + void delete(long id) { + try { + deleteStmt.setLong(1,id); + int n = deleteStmt.executeUpdate(); + if( n==0 ) { + logger.error("delete not found for id="+id); + } + } catch(SQLException e) { + logger.error("update failed",e); + } + } + + void begin() { + try { + if( trans++ == 0 ) + con.setAutoCommit(false); + } catch(SQLException e) { + logger.error("begin failed",e); + } + } + + void commit() { + try { + if( trans <= 0 ) { + logger.error("commit not in transaction"); + return; + } + if( --trans == 0 ) + con.setAutoCommit(true); + } catch(SQLException e) { + logger.error("begin failed",e); + } + } + + void rollback() { + try { + if( --trans != 0 ) { + logger.error("rollback failed trans="+trans); + return; + } + con.rollback(); + con.setAutoCommit(true); + } catch(SQLException e) { + logger.error("begin failed",e); + } + } + +}