############################################################################## # # Copyright (c) 2004 TINY SPRL. (http://tiny.be) All Rights Reserved. # # $Id: stock.py 1005 2005-07-25 08:41:42Z nicoe $ # # WARNING: This program as such is intended to be used by professional # programmers who take the whole responsability of assessing all potential # consequences resulting from its eventual inadequacies and bugs # End users who are looking for a ready-to-use solution with commercial # garantees and support are strongly adviced to contract a Free Software # Service Company # # This program is Free Software; you can redistribute it and/or # modify it under the terms of the GNU General Public License # as published by the Free Software Foundation; either version 2 # of the License, or (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. # ############################################################################## import datetime import time import netsvc from osv import fields,osv import ir #---------------------------------------------------------- # Incoterms #---------------------------------------------------------- class stock_incoterms(osv.osv): _name = "stock.incoterms" _description = "Incoterms" _columns = { 'name': fields.char('Name', size=64, required=True), 'code': fields.char('Code', size=3, required=True), 'active': fields.boolean('Active'), } _defaults = { 'active': lambda *a: True, } stock_incoterms() class stock_lot(osv.osv): _name = "stock.lot" _description = "Lot" _columns = { 'name': fields.char('Lot Name', size=64, required=True), 'active': fields.boolean('Active'), 'tracking': fields.char('Tracking', size=64), 'move_ids': fields.one2many('stock.move', 'lot_id', 'Move lines'), } _defaults = { 'active': lambda *a: True, } stock_lot() #---------------------------------------------------------- # Stock Location #---------------------------------------------------------- class stock_location(osv.osv): _name = "stock.location" _description = "Location" _columns = { 'name': fields.char('Location Name', size=64, required=True), 'active': fields.boolean('Active'), 'usage': fields.selection([('supplier','Supplier Location'),('internal','Internal Location'),('customer','Customer Location'),('inventory','Inventory'),('procurement','Procurement'),('production','Production')], 'Location type'), 'allocation_method': fields.selection([('fifo','FIFO'),('lifo','LIFO'),('nearest','Nearest')], 'Allocation Method', required=True), 'account_id': fields.many2one('account.account', string='Inventory Account', domain=[('type','=','stock_inventory')]), 'location_id' : fields.many2one('stock.location', 'Parent Location'), 'child_ids': fields.one2many('stock.location', 'location_id', 'Contains'), 'comment': fields.text('Additional Information'), 'posx': fields.integer('Position X', required=True), 'posy': fields.integer('Position Y', required=True), 'posz': fields.integer('Position Z', required=True) } _defaults = { 'active': lambda *a: 1, 'usage': lambda *a: 'internal', 'allocation_method': lambda *a: 'fifo', 'posx': lambda *a: 0, 'posy': lambda *a: 0, 'posz': lambda *a: 0, } def _product_get_all_report(self, cr, uid, ids, product_ids=False, context={}): return self._product_get_report(cr, uid, ids, product_ids, context, recursive=True) def _product_get_report(self, cr, uid, ids, product_ids=False, context={}, recursive=False): if not product_ids: product_ids = self.pool.get('product.product').search(cr, uid, []) result = [] for id in ids: for prod_id in product_ids: product = self.pool.get('product.product').browse(cr, uid, [prod_id])[0] fnc = self._product_get if recursive: fnc = self._product_all_get qty = fnc(cr, uid, id, [prod_id], {'uom': product.uom_id.id}) if qty[prod_id]: result.append({ 'price': product.list_price, 'name': product.name, 'variants': product.variants or '', 'uom': product.uom_id.name, 'amount':qty[prod_id] }) return result def _product_get(self, cr, uid, id, product_ids=False, context={}, states=['done']): states_str = ','.join(map(lambda x:"'"+x+"'", states)) if not product_ids: product_ids = self.pool.get('product.product').search(cr, uid, []) prod_ids_str = ','.join(map(str,product_ids)) cr.execute('select sum(product_qty),product_id,product_uom from stock_move where location_dest_id=%d and product_id in ('+prod_ids_str+') and state in ('+states_str+') group by product_id,product_uom', (id,)) results = cr.fetchall() cr.execute('select -sum(product_qty),product_id,product_uom from stock_move where location_id=%d and product_id in ('+prod_ids_str+') and state in ('+states_str+') group by product_id,product_uom', (id,)) results += cr.fetchall() res = {} for id in product_ids: res[id] = 0.0 for (amount, prod_id, prod_uom) in results: amount = self.pool.get('product.uom')._compute_qty(cr, uid, prod_uom, amount, context.get('uom',False)) res[prod_id]+=amount return res def _product_all_get(self, cr, uid, id, product_ids=False, context={}, states=['done']): result = self._product_get(cr, uid, id, product_ids, context, states) childs = self.read(cr, uid, [id], ['child_ids'])[0]['child_ids'] for child in childs: result2 = self._product_all_get(cr, uid, child, product_ids, context, states) for (key,val) in result2.items(): result.setdefault(key, 0.0) result[key]+=val return result def _product_virtual_get(self, cr, uid, id, product_ids=False, context={}, states=['done']): return self._product_all_get(cr, uid, id, product_ids, context, ['confirmed','waiting','assigned','done']) # # TODO: Different method for location search ! # def _location_order_get(self, cr, uid, ids, context): ids2 = ids[:] datas = self.read(cr, uid, ids, ['child_ids']) for data in datas: ids2 += self._location_order_get(cr, uid, data['child_ids'], context) return ids2 # # TODO: # Improve this function # # Returns: # [ (tracking_id, product_qty, location_id) ] # def _product_reserve(self, cr, uid, ids, product_id, product_qty, context={}): result = [] amount = 0.0 for id in self._location_order_get(cr, uid, ids, context): cr.execute("select product_uom,sum(product_qty) as product_qty from stock_move where location_dest_id=%d and product_id=%d and state='done' group by product_uom", (id,product_id)) results = cr.dictfetchall() cr.execute("select product_uom,-sum(product_qty) as product_qty from stock_move where location_id=%d and product_id=%d and state in ('done', 'assigned') group by product_uom", (id,product_id)) results += cr.dictfetchall() total = 0.0 results2 = 0.0 for r in results: amount = self.pool.get('product.uom')._compute_qty(cr, uid, r['product_uom'],r['product_qty'], context.get('uom',False)) results2 += amount total += amount if total<=0.0: continue amount = results2 if amount>0: if amount>min(total,product_qty): amount = min(product_qty,total) result.append((amount,id)) product_qty -= amount total -= amount if product_qty<=0.0: return result if total<=0.0: continue return False stock_location() #---------------------------------------------------------- # Stock Move #---------------------------------------------------------- class stock_move_lot(osv.osv): _name = "stock.move.lot" _description = "Move Lot" _columns = { 'name': fields.char('Move Description', size=64, required=True), 'active': fields.boolean('Active'), 'state': fields.selection( (('draft','Draft'),('done','Moved')), 'State', readonly=True), 'serial': fields.char('Tracking Number', size=32), 'date_planned': fields.date('Planned Date'), 'date_moved': fields.date('Actual Date'), 'lot_id': fields.many2one('stock.lot','Lot', required=True), 'loc_dest_id': fields.many2one('stock.location', 'Destination Location', required=True), 'address_id': fields.many2one('res.partner.address', 'Destination Address'), 'origin': fields.char('Origin', size=64), } _defaults = { 'active': lambda *a: 1, 'state': lambda *a: 'draft', 'date_planned': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'), } # # TODO: test if valid # def action_move(self, cr, uid, ids, context={}): for move in self.browse(cr, uid, ids, context): lot_remove = [] for m in move.lot_id.move_ids: new_id = self.pool.get('stock.move').copy(cr, uid, m.id, {'location_id': m.location_dest_id.id, 'location_dest_id': move.loc_dest_id.id, 'date_moved': time.strftime('%Y-%m-%d'), 'state':'done', 'picking_id': False}) cr.execute('insert into stock_move_history_ids (parent_id,child_id) values (%d,%d)', (m.id, new_id)) lot_remove.append(m.id) self.pool.get('stock.move').write(cr, uid, lot_remove, {'lot_id':False}) self.write(cr,uid, ids, {'state':'done'}) return True stock_move_lot() class stock_tracking(osv.osv): _name = "stock.tracking" _description = "Stock Tracking Lots" def checksum(sscc): salt = '31' * 8 + '3' sum = 0 for sscc_part, salt_part in zip(sscc, salt): sum += int(sscc_part) * int(salt_part) return (10 - (sum % 10)) % 10 checksum = staticmethod(checksum) def make_sscc(self, cr, uid, context={}): sequence = self.pool.get('ir.sequence').get(cr, uid, 'stock.lot.tracking') print sequence return sequence + str(self.checksum(sequence)) _columns = { 'name': fields.char('Tracking', size=64, required=True), 'active': fields.boolean('Active'), 'serial': fields.char('Reference', size=64), 'move_ids' : fields.one2many('stock.move', 'tracking_id', 'Moves tracked'), 'date': fields.datetime('Date create', required=True), } _defaults = { 'active': lambda *a: 1, # 'name': lambda x,y,z,c: x.pool.get('ir.sequence').get(y,z,'stock.lot.tracking'), 'name' : make_sscc, 'date': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'), } #_sql = 'ALTER TABLE stock_tracking ADD CONSTRAINT stock_lot_tracking_unique UNIQUE (name,serial)' def name_search(self, cr, user, name, args=[], operator='ilike', context={}): ids = self.search(cr, user, [('serial','=',name)]+ args) ids += self.search(cr, user, [('name',operator,name)]+ args) return self.name_get(cr, user, ids) def name_get(self, cr, uid, ids, context={}): if not len(ids): return [] res = [(r['id'], r['name']+' ['+(r['serial'] or '')+']') for r in self.read(cr, uid, ids, ['name','serial'], context)] return res def unlink(self, cr ,uid, ids): raise 'You can not remove a lot line !' stock_tracking() #---------------------------------------------------------- # Stock Picking #---------------------------------------------------------- class stock_picking(osv.osv): _name = "stock.picking" _description = "Picking list" _columns = { 'name': fields.char('Picking Name', size=64, required=True), 'origin': fields.char('Origin', size=64), 'type': fields.selection([('out','Sending Goods'),('in','Getting Goods'),('internal','Internal')], 'Shipping Type', required=True), 'active': fields.boolean('Active'), 'note': fields.text('Notes'), 'location_id': fields.many2one('stock.location', 'Location'), 'location_dest_id': fields.many2one('stock.location', 'Dest. Location'), 'move_type': fields.selection([('direct','Direct Delivery'),('one','All at once')],'Delivery Method', required=True), 'state': fields.selection([ ('draft','draft'), ('auto','waiting'), ('confirmed','confirmed'), ('assigned','assigned'), ('done','done'), ('cancel','cancel'), ], 'State', readonly=True), 'date':fields.datetime('Date create'), 'move_lines': fields.one2many('stock.move', 'picking_id', 'Move Lines', states={'done':[('readonly',True)], 'cancel':[('readonly',True)]}), 'auto_picking': fields.boolean('Auto-Picking'), 'work': fields.boolean('Work todo'), 'loc_move_id': fields.many2one('stock.location', 'Move to Location'), 'address_id': fields.many2one('res.partner.address', 'Partner Address'), 'lot_id': fields.many2one('stock.lot', 'Consumer Lot Created'), 'move_lot_id': fields.many2one('stock.move.lot', 'Moves Created') } _defaults = { 'work': lambda *a: 0, 'active': lambda *a: 1, 'state': lambda *a: 'draft', 'move_type': lambda *a: 'direct', 'type': lambda *a: 'in', 'date': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'), } def action_confirm(self, cr, uid, ids, *args): self.write(cr,uid,ids, {'state':'confirmed'}) todo = [] for line in self.browse(cr,uid, ids): for r in line.move_lines: if r.state=='draft': todo.append(r.id) if len(todo): self.pool.get('stock.move').action_confirm(cr,uid, todo) return True def test_auto_picking(self, cr, uid, ids): # TODO: Check locations to see if in the same location ? return True def action_assign(self, cr, uid, ids, *args): for pick in self.browse(cr, uid, ids): move_ids = [x.id for x in pick.move_lines if x.state=='confirmed'] self.pool.get('stock.move').action_assign(cr, uid, move_ids) return True def force_assign(self, cr, uid, ids, *args): wf_service = netsvc.LocalService("workflow") for pick in self.browse(cr, uid, ids): move_ids = [x.id for x in pick.move_lines if x.state == 'confirmed'] self.pool.get('stock.move').force_assign(cr, uid, move_ids) wf_service.trg_write(uid, 'stock.picking', pick.id, cr) return True def action_assign_wkf(self, cr, uid, ids): self.write(cr, uid, ids, {'state':'assigned'}) return True def test_finnished(self, cr, uid, ids): for id in ids: cr.execute('select id,state from stock_move where picking_id=%d', (id,)) for x in cr.fetchall(): if x[1] not in ('done','cancel'): return False return True def test_assigned(self, cr, uid, ids): ok = True for pick in self.browse(cr, uid, ids): mt = pick.move_type for move in pick.move_lines: if (move.state in ('confirmed','draft')) and (mt=='one'): return False if (mt=='direct') and move.state=='assigned': return True ok = ok and (move.state in ('cancel','done','assigned')) return ok def action_cancel(self, cr, uid, ids): for pick in self.browse(cr, uid, ids): ids2 = [move.id for move in pick.move_lines] self.pool.get('stock.move').action_cancel(cr, uid, ids2) self.write(cr,uid, ids, {'state':'cancel'}) return True # # TODO: change and create a move if not parents # def action_done(self, cr, uid, ids, *args): for pick in self.browse(cr, uid, ids): if pick.move_type=='one' and pick.loc_move_id: if pick.lot_id: id = self.pool.get('stock.move.lot').create(cr, uid, { 'name': 'MOVE:'+pick.name, 'origin': 'PICK:'+str(pick.id), 'lot_id': pick.lot_id.id, 'loc_dest_id': pick.loc_move_id.id, 'address_id': pick.address_id.id }) self.write(cr, uid, [pick.id], {'move_lot_id':id}) self.write(cr,uid, ids, {'state':'done'}) return True def action_move(self, cr, uid, ids, context={}): for pick in self.browse(cr, uid, ids): todo = [] for move in pick.move_lines: if move.state=='assigned': todo.append(move.id) if len(todo): self.pool.get('stock.move').action_done(cr, uid, todo) lot_id = self.pool.get('stock.lot').create(cr, uid, {'name':'PICK:'+pick.name}) self.pool.get('stock.move').write(cr,uid, todo, {'lot_id':lot_id}) self.write(cr, uid, [pick.id], {'lot_id':lot_id}) if pick.move_type=='direct' and pick.loc_move_id: id = self.pool.get('stock.move.lot').create(cr, uid, { 'name': 'MOVE:'+pick.name, 'origin': 'PICK:'+str(pick.id), 'lot_id': lot_id, 'loc_dest_id': pick.loc_move_id.id, 'address_id': pick.address_id.id }) self.write(cr, uid, [pick.id], {'move_lot_id':id}) return True stock_picking() class stock_production_lot(osv.osv): _name = 'stock.production.lot' _description = 'Production lot' _columns = { 'name' : fields.char('Code', size=15, required=True), 'date': fields.datetime('Date create', required=True), } _defaults = { 'date': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'), } stock_production_lot() # ---------------------------------------------------- # Move # ---------------------------------------------------- # # Fields: # location_dest_id is only used for predicting futur stocks # class stock_move(osv.osv): def _getSSCC(self, cr, uid, context={}): cr.execute('select id from stock_tracking where create_uid=%d order by id desc limit 1', (uid,)) res = cr.fetchone() return (res and res[0]) or False _name = "stock.move" _description = "Stock Move" _columns = { 'name': fields.char('Name', size=64, required=True), 'priority': fields.selection([('0','Not urgent'),('1','Urgent')], 'Priority'), 'date': fields.datetime('Date Created'), 'date_planned': fields.date('Planned date', required=True), 'product_id': fields.many2one('product.product', 'Product', required=True ), 'product_qty': fields.float('Quantity (UOM)', required=True), 'product_uom': fields.many2one('product.uom', 'Product UOM', required=True), 'product_uos_qty': fields.float('Quantity (UOS)'), 'product_uos': fields.many2one('product.uom', 'Product UOS'), 'product_packaging' : fields.many2one('product.packaging', 'Product packaging'), 'location_id': fields.many2one('stock.location', 'Location',required=True), 'location_dest_id': fields.many2one('stock.location', 'Dest. Location',required=True), 'address_id' : fields.many2one('res.partner.address', 'Dest. Address', required=False), 'prodlot_id' : fields.many2one('stock.production.lot', 'Production lot'), 'tracking_id': fields.many2one('stock.tracking', 'Lot Tracking'), 'lot_id': fields.many2one('stock.lot', 'Lot'), 'move_dest_id': fields.many2one('stock.move', 'Dest. Move'), 'move_history_ids': fields.many2many('stock.move', 'stock_move_history_ids', 'parent_id', 'child_id', 'Move History'), 'move_history_ids2': fields.many2many('stock.move', 'stock_move_history_ids', 'child_id', 'parent_id', 'Move History'), 'picking_id': fields.many2one('stock.picking', 'Picking list'), 'state': fields.selection([('draft','Draft'),('waiting','Waiting'),('confirmed','Confirmed'),('assigned','Assigned'),('done','Done'),('cancel','cancel')], 'State', readonly=True), } _defaults = { 'state': lambda *a: 'draft', 'priority': lambda *a: '1', #'tracking_id' : _getSSCC, 'product_qty': lambda *a: 1.0, 'date_planned': lambda *a: time.strftime('%Y-%m-%d'), 'date': lambda *a: time.strftime('%Y-%m-%d'), } def onchange_product_id(self, cr, uid, context, prod_id=False, loc_id=False, loc_dest_id=False): if not prod_id: return {} product = self.pool.get('product.product').browse(cr, uid, [prod_id])[0] result = { 'name': product.name, 'product_uom': product.uom_id.id, } if loc_id: result['location_id'] = loc_id if loc_dest_id: result['location_dest_id'] = loc_dest_id return {'value':result} def action_confirm(self, cr, uid, ids, context={}): self.write(cr, uid, ids, {'state':'confirmed'}) return True def action_assign(self, cr, uid, ids, *args): todo = [] for move in self.browse(cr, uid, ids): if move.state in ('confirmed','waiting'): todo.append(move.id) res = self.check_assign(cr, uid, todo) return res def force_assign(self, cr, uid, ids, context={}): self.write(cr, uid, ids, {'state' : 'assigned'}) return True # # Duplicate stock.move # def check_assign(self, cr, uid, ids, context={}): done = [] pickings = {} for move in self.browse(cr, uid, ids): if move.state in ('confirmed','waiting'): res = self.pool.get('stock.location')._product_reserve(cr, uid, [move.location_id.id], move.product_id.id, move.product_qty, {'uom': move.product_uom.id}) if res: done.append(move.id) pickings[move.picking_id.id] = 1 r = res.pop(0) cr.execute('update stock_move set location_id=%d, product_qty=%f where id=%d', (r[1],r[0], move.id)) while len(res): r = res.pop(0) move_id = self.copy(cr, uid, move.id, {'product_qty':r[0], 'location_id':r[1]}) done.append(move_id) #cr.execute('insert into stock_move_history_ids values (%d,%d)', (move.id,move_id)) if len(done): self.write(cr, uid, done, {'state':'assigned'}) for pick_id in pickings: wf_service = netsvc.LocalService("workflow") wf_service.trg_write(uid, 'stock.picking', pick_id, cr) return len(done) # # Cancel move => cancel others move and pickings # def action_cancel(self, cr, uid, ids): if not len(ids): return True pickings = {} for move in self.browse(cr, uid, ids): if move.state in ('confirmed','waiting','assigned','draft'): if move.picking_id: pickings[move.picking_id.id] = True self.write(cr, uid, ids, {'state':'cancel'}) ids_lst = ','.join(map(str,ids)) for pick_id in pickings: wf_service = netsvc.LocalService("workflow") wf_service.trg_validate(uid, 'stock.picking', pick_id, 'button_cancel', cr) ids2 = [] for res in self.read(cr, uid, ids, ['move_dest_id']): if res['move_dest_id']: ids2.append(res['move_dest_id'][0]) self.action_cancel(cr,uid, ids2) return True def action_done(self, cr, uid, ids, *args): for move in self.browse(cr, uid, ids): if move.move_dest_id.id and (move.state != 'done'): mid = move.move_dest_id.id if move.move_dest_id.id: cr.execute('insert into stock_move_history_ids (parent_id,child_id) values (%d,%d)', (move.id, move.move_dest_id.id)) if move.move_dest_id.state in ('waiting','confirmed'): self.write(cr, uid, [move.move_dest_id.id], {'state':'assigned'}) if move.move_dest_id.picking_id: wf_service = netsvc.LocalService("workflow") wf_service.trg_write(uid, 'stock.picking', move.move_dest_id.picking_id.id, cr) else: pass # self.action_done(cr, uid, [move.move_dest_id.id]) # # Accounting Entries # # acc_src = None # acc_dest = None # if move.location_id.account_id: # acc_src = move.location_id.account_id.id # if move.location_dest_id.account_id: # acc_dest = move.location_dest_id.account_id.id # if acc_src or acc_dest: # test = [('product.product', move.product_id.id)] # if move.product_id.categ_id: # test.append( ('product.category', move.product_id.categ_id.id) ) # if not acc_src: # acc_src = ir.ir_get(cr,uid,'meta','account.expense',test)[0][2] # if not acc_dest: # acc_dest = ir.ir_get(cr,uid,'meta','account.expense',test)[0][2] # if acc_src != acc_dest: # amount = move.product_qty * move.product_id.standard_price # lines = [ # (0,0,{'name': 'Stock', 'amount': -amount, 'account_id': acc_src}), # (0,0,{'name': 'Stock', 'amount': amount, 'account_id': acc_dest}) # ] # self.pool.get('account.move').create(cr, uid, {'name': 'Stock', 'type':'undefined', 'line_id': lines}) self.write(cr, uid, ids, {'state':'done'}) return True stock_move() class stock_inventory(osv.osv): _name = "stock.inventory" _description = "Inventory" _columns = { 'name': fields.char('Inventory', size=64, required=True), 'date': fields.datetime('Date create', required=True), 'date_done': fields.datetime('Date done'), 'inventory_line_id': fields.one2many('stock.inventory.line', 'inventory_id', 'Inventories'), 'move_ids': fields.many2many('stock.move', 'stock_inventory_move_rel', 'inventory_id', 'move_id', 'Created Moves'), 'state': fields.selection( (('draft','Draft'),('done','Done')), 'State', readonly=True), } _defaults = { 'date': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'), 'state': lambda *a: 'draft', } # # Update to support tracking # def action_done(self, cr, uid, ids, *args): for inv in self.browse(cr,uid,ids): move_ids = [] move_line=[] for line in inv.inventory_line_id: pid=line.product_id.id price=line.product_id.standard_price or 0.0 amount=self.pool.get('stock.location')._product_get(cr, uid, line.location_id.id, [pid], {'uom': line.product_uom.id})[pid] change=line.product_qty-amount if change: location_id = ir.ir_get(cr,uid,'meta','stock.lot.inventory',[('product.product', pid)])[0][2] value = { 'name': 'INV:'+str(line.inventory_id.id)+':'+line.inventory_id.name, 'product_id': line.product_id.id, 'product_uom': line.product_uom.id, 'date': inv.date, 'date_planned': inv.date, 'state': 'assigned' } if change>0: value.update( { 'product_qty': change, 'location_id': location_id, 'location_dest_id': line.location_id.id, }) else: value.update( { 'product_qty': -change, 'location_id': line.location_id.id, 'location_dest_id': location_id, }) move_ids.append(self.pool.get('stock.move').create(cr, uid, value)) if len(move_ids): self.pool.get('stock.move').action_done(cr, uid, move_ids) self.write(cr, uid, [inv.id], {'state':'done', 'date_done': time.strftime('%Y-%m-%d %H:%M:%S'), 'move_ids': [(6,0,move_ids)]}) return True def action_cancel(self, cr, uid, ids, context={}): for inv in self.browse(cr,uid,ids): self.pool.get('stock.move').action_cancel(cr, uid, [x.id for x in inv.move_ids]) self.write(cr, uid, [inv.id], {'state':'draft'}) return True stock_inventory() class stock_inventory_line(osv.osv): _name = "stock.inventory.line" _description = "Inventory line" _columns = { 'inventory_id': fields.many2one('stock.inventory','Inventory', ondelete='cascade'), 'location_id': fields.many2one('stock.location','Location', required=True), 'product_id': fields.many2one('product.product', 'Product', required=True ), 'product_uom': fields.many2one('product.uom', 'Product UOM', required=True ), 'product_qty': fields.float('Quantity') } def on_change_product_id(self, cr, uid, ids, location_id, product, uom=False): if not product: return {} if not uom: prod = self.pool.get('product.product').browse(cr, uid, [product], {'uom': uom})[0] uom = prod.uom_id.id amount=self.pool.get('stock.location')._product_get(cr, uid, location_id, [product], {'uom': uom})[product] result = {'product_qty':amount, 'product_uom':uom} return {'value':result} stock_inventory_line() #---------------------------------------------------------- # Stock Warehouse #---------------------------------------------------------- class stock_warehouse(osv.osv): _name = "stock.warehouse" _description = "Warehouse" _columns = { 'name': fields.char('Name', size=60, required=True), # 'partner_id': fields.many2one('res.partner', 'Owner'), # 'partner_address_id': fields.many2one('res.partner.address', 'Owner Address'), 'lot_input_id': fields.many2one('stock.location', 'Location Input', required=True ), 'lot_stock_id': fields.many2one('stock.location', 'Location Stock', required=True ), 'lot_output_id': fields.many2one('stock.location', 'Location Output', required=True ), } stock_warehouse()